Data preparation script for Sierra Nevada time series analysis

# 1. Read in isotope data from various sources
# First load helper functions 'read_jena_ams_results.R', 'read_jena_iso_results.R' 
source("./utilities/jena_ams_ingest.R")
source("./utilities/jena_iso_ingest.R")
source("./utilities/jena_elm_ingest.R")
# color palettes for ECO & PM
warm <- "#BF812D"
cool <- "#80CDC1"
cold <- "#01665E"
granite <- "#9daba9"
andesite <- "#382dbf"
basalt <- "#bf382d"
# Create template for bulk soil data
template19.fx <- function(pm, eco, ndepth) {
  df <- data.frame(Year = rep(2019, ndepth * 3),
                   PM = rep(pm, ndepth * 3),
                   ECO = rep(eco, ndepth * 3),
                   pro_rep = rep(seq(1,3), each = ndepth),
                   lyr_top = rep(seq(0, (ndepth-1) * 10, by = 10), 3),
                   lyr_bot = rep(seq(10, (ndepth) * 10, by = 10), 3))
  df$pro_name <- paste0(df$PM, df$ECO, "_", df$pro_rep)
  df$lyr_name <- paste0(df$pro_name, "_", df$lyr_top, "-", df$lyr_bot)
  return(df)
}

# Create template for composite soil data (incubations, density fractions, etc.)
template.comp.fx <- function(year, pm, eco, depth_bot = c(10, 20, 30), dat) {
  ndepth <- length(depth_bot)
  df <- data.frame(Year = rep(year, ndepth * length(pm)),
                   PM = rep(pm, each = ndepth * length(eco)),
                   ECO = rep(eco, each = ndepth))
  df$lyr_bot <- depth_bot
  df$lyr_top <- sapply(seq_along(depth_bot), function(i) {
    if (i == 1) {
      depth_top <- 0
      } else {
        depth_top <- depth_bot[i - 1]
      }
  })
  df$pro_name <- paste0(df$PM, df$ECO, "_comp")
  n <- nrow(df)
  if (dat == "inc") {
    df <- rbind(df, df)
    df$rep <- rep(c("a", "b"), each = n)
    df$lyr_name <- paste0(df$pro_name, "_", 
                          df$lyr_top, "-", 
                          df$lyr_bot, "_", 
                          df$Year, "_",
                          df$rep)
  } else if (dat == "density") {
    df <- rbind(df, df, df)
    df$frc <- rep(c("fLF", "oLF", "mnC"), each = n)
    df$lyr_name <- paste0(df$pro_name, "_", 
                          df$lyr_top, "-", 
                          df$lyr_bot, "_", 
                          df$Year, "_",
                          df$frc)
  }
  return(df)
}

# templates for bulk soil data
# GRrf 
GRrf <- template19.fx("GR", "rf", 7)
GRrf <- if(any(GRrf$lyr_name == "GRrf_1_60_70")) {
  GRrf <- GRrf[-which(GRrf$lyr_name == "GRrf_1_60_70"), ] # NB: GRrf_1_60_70 doesn't exist
} else {
  GRrf <- GRrf
}
# GRwf
GRwf <- template19.fx("GR", "wf", 9)
# GRpp
GRpp <- template19.fx("GR", "pp", 8)

# ANrf 
ANrf <- template19.fx("AN", "rf", 6)
# ANwf
ANwf <- template19.fx("AN", "wf", 6)
# ANpp
ANpp <- template19.fx("AN", "pp", 8)

# BSrf 
BSrf <- template19.fx("BS", "rf", 8)
BSrf <- if(any(BSrf$lyr_name == "GRrf_1_60_70")) {
  BSrf <- BSrf[-which(BSrf$lyr_name == "BSrf_1_70_80"), ] # NB: BSrf_1_70_80 doesn't exist
} else {
  BSrf <- BSrf
} 
# BSwf
BSwf <- template19.fx("BS", "wf", 7)
# BSpp
BSpp <- template19.fx("BS", "pp", 8)
BSpp[BSpp$lyr_bot == 80, "lyr_bot"] <- 75 # only sampled to 75cm, not 80

sra.2019.df <- rbind(GRrf, GRwf, GRpp,
                     ANrf, ANwf, ANpp,
                     BSrf, BSwf, BSpp)

# template for 2019 incubation data
sra.2019.inc.df <- template.comp.fx(2019, 
                                    pm = c("AN", "BS", "GR"),
                                    eco = c("pp", "wf", "rf"),
                                    dat = "inc")

## template for 2001 incubation data
# list of depths for 2001 inc samples
depth_bot_2001.ls <- list(ANpp = c(6, 13, 33),
                          ANwf = c(11, 35),
                          ANrf = c(11, 32),
                          BSpp = c(7, 18, 28),
                          BSwf = c(10, 19),
                          BSrf = c(8, 15, 30),
                          GRpp = c(7, 15, 27),
                          GRwf = c(4, 13, 28),
                          GRrf = c(8, 27)) 
# template for inputs to template.comp.fx (year, pm, eco)
inc.2001.template <- lapply(seq_along(depth_bot_2001.ls), function(i) {
  nms <- names(depth_bot_2001.ls)
  ls <- list(year = 2001, 
             pm = substr(nms[i], 1, 2), 
             eco = substr(nms[i], 3, 4))
  ls$depth_bot <- depth_bot_2001.ls[[i]]
  return(ls)
})
# create template data frame by iteratively calling template.comp.fx
sra.2001.inc.df <- bind_rows(
  lapply(seq_along(inc.2001.template), function(i) {
    template.comp.fx(year = inc.2001.template[[i]][[1]],
                     pm = inc.2001.template[[i]][[2]],
                     eco = inc.2001.template[[i]][[3]],
                     depth_bot = inc.2001.template[[i]][[4]],
                     dat = "inc")
  })
)

# 2001 bulk soil template
sra.2001 <- vector(mode = "list", length = length(unique(sra.2019.df$pro_name)))
names(sra.2001) <- unique(sra.2019.df$pro_name)

# 2019 bulk soil template
sra.2019 <- sra.2001

# inc templates for merging 14C data
sra.2019.inc <- vector(mode = "list", length = length(unique(sra.2019.inc.df$pro_name)))
names(sra.2019.inc) <- unique(sra.2019.inc.df$pro_name)
sra.2001.inc <- sra.2019.inc
# copies for reps of incubations
sra.2019.inc_L <- sra.2019.inc
names(sra.2019.inc_L) <- substr(names(sra.2019.inc_L), 1, 4)
# complete cases, convert type for calculating stocks later
# could calculate stocks now and then remove for the following steps where not needed

## 2001 summary data
soc.2001 <- data.frame(read_excel("../data/external/sra_ras_sum/sierra_data_summary_2020.xlsx",
                                  sheet = "2001_bulk_data"))

# create list; remove BS samples deeper than 30 cm
soc.2001.ls <- lapply(split(soc.2001, soc.2001$PMeco), function(df) {
  df <- type.convert(df[complete.cases(df), c("ID", "C.", "mass_kgm2", "PMeco", "pro_rep", "lyr_top", "lyr_bot")])
 return(df[which(df$lyr_bot < 36), ])
})

# Incubation samples combined 0-3 and 3-8 depth increments for BSrf and GRrf
# combine BSrf and GRrf initial depths
# function for calculating weighted average of first two depth increment C content
d1d2.fx <- function(df) {
  d1d2 <- data.frame(ID = paste(df$PMeco[1], df$pro_rep[1], df$lyr_top[1], df$lyr_bot[2], sep = "_"),
                     C. = sum(df$C.[1] * ((df$lyr_bot[1] - df$lyr_top[1]) / df$lyr_bot[2]), df$C.[2] * ((df$lyr_bot[2] - df$lyr_top[2]) / df$lyr_bot[2])),
                     mass_kgm2 = sum(df$mass_kgm2[1], df$mass_kgm2[2]),
                     PMeco = df$PMeco[1],
                     pro_rep = df$pro_rep[1],
                     lyr_top = df$lyr_top[1],
                     lyr_bot = df$lyr_bot[2])
  return(rbind(d1d2,
               df[3:nrow(df), ]))
}
# Run d1d2.fx for BSrf, GRrf
soc.2001.ls.inc <- soc.2001.ls 
soc.2001.ls.inc$GRrf <- bind_rows(lapply(split(soc.2001.ls$GRrf, soc.2001.ls$GRrf$pro_rep), d1d2.fx))
soc.2001.ls.inc$BSrf <- bind_rows(lapply(split(soc.2001.ls$BSrf, soc.2001.ls$BSrf$pro_rep), d1d2.fx))

# summarize inc
soc.2001.inc.sum <- data.frame(bind_rows(lapply(soc.2001.ls.inc, function(df) {
  df %>%
    mutate(ID2 = paste0(PMeco, "_", lyr_top, "-", lyr_bot)) %>%
    group_by(ID2, PMeco, lyr_top, lyr_bot) %>%
    summarize(c_pct_avg = mean(C.))
})))
save(soc.2001.inc.sum, file = "soc.2001.inc.sum.RData")

# calculate SOC stocks
soc.2001.ls <- lapply(soc.2001.ls, function(df) {
  df$lyr_soc_kgm2 <- df$C. * df$mass_kgm2 * 10^-2
  return(df)
})

# summarize [note that soc stocks are dropped]
soc.2001.sum <- data.frame(bind_rows(lapply(soc.2001.ls, function(df) {
  df %>%
    mutate(ID2 = paste0(PMeco, "_", lyr_top, "-", lyr_bot)) %>%
    group_by(ID2, PMeco, lyr_top, lyr_bot) %>%
    summarize(c_pct_avg = mean(C.))
})))

# 2019 data
sra.2019.cn.sum <- data.frame(
  bind_rows(unlist(elm_results_ls, recursive = FALSE)) %>%
  mutate(PMeco = sapply(strsplit(ID, "_"), "[", 2),
         depth = sapply(strsplit(ID, "_"), "[", 4)) %>%
  group_by(PMeco, depth) %>%
  summarize(across(c(C, N), .fns = mean))) %>%
  rename(c_pct_avg = C,
         n_pct_avg = N)
sra.2019.cn.sum$ID2 <- paste(sra.2019.cn.sum$PMeco, sra.2019.cn.sum$depth, sep = "_")
save(sra.2019.cn.sum, file = "sra.2019.cn.sum.RData")

Merge templates with 14C, C, and N data

Radiocarbon analyses for the 2001 samples were not run originally, but were completed on archived samples in 2020.

# Extract 14C data for 2001 samples
ams_results_ls_S01 <- ams_results_ls[grep("S01", names(ams_results_ls))]
for(i in seq_along(sra.2001)) {
  sra.2001[[i]] <- lapply(ams_results_ls_S01, function(ls) {
    lapply(ls, function(df) {
      if(any(grepl(names(sra.2001)[i], df$Probe))) {
       df[grep(names(sra.2001)[i], df$Probe), ] 
      }
    })
  })
  sra.2001[[i]] <- Filter(Negate(is.null), unlist(sra.2001[[i]], recursive = FALSE))
}
sra.2001 <- bind_rows(unlist(sra.2001, recursive = FALSE))

# create ID field, trim df, and add depths
sra.2001$ID <- unlist(strsplit(sra.2001$Probe, "_Sierra Nevada_2001"))
sra.2001 <- sra.2001[ , c("ID", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2001) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
sra.2001$lyr_top <- as.numeric(ifelse(substr(sra.2001$ID, 9, 9) == "-",
                                      substr(sra.2001$ID, 8, 8),
                                      substr(sra.2001$ID, 8, 9)))
sra.2001$lyr_bot <- as.numeric(ifelse(substr(sra.2001$ID, 9, 9) == "-", 
                                      substr(sra.2001$ID, 10, nchar(sra.2001$ID)),
                                      substr(sra.2001$ID, 11, nchar(sra.2001$ID))))
sra.2001$pro_rep <- substr(sra.2001$ID, 6, 6)
sra.2001$PM <- factor(substr(sra.2001$ID, 1, 2))
sra.2001$ECO <- factor(substr(sra.2001$ID, 3, 4), levels = c("pp", "wf", "rf"))
sra.2001$pro_name <- substr(sra.2001$ID, 1, 6)
sra.2001$PMeco <- substr(sra.2001$ID, 1, 4)

# remove outlier ANpp sample
sra.2001 <- sra.2001[-which(sra.2001$ID == "ANpp_3_6-13"), ]

# make list by PMeco
sra.2001.ls <- split(sra.2001, sra.2001$PMeco)
# Extract 14C data for 2019 samples
ams_results_ls_S19 <- ams_results_ls[grep("soil-S19", names(ams_results_ls))]
for(i in seq_along(sra.2019)) {
  sra.2019[[i]] <- lapply(ams_results_ls_S19, function(ls) {
    lapply(ls, function(df) {
      if(any(grepl(names(sra.2019)[i], df$Probe))) {
       df[grep(names(sra.2019)[i], df$Probe), ] 
      }
    })
  })
  sra.2019[[i]] <- Filter(Negate(is.null), unlist(sra.2019[[i]], recursive = FALSE))
}
sra.2019 <- bind_rows(unlist(sra.2019, recursive = FALSE))

## merge w/ 2019 template
# rename cols in AMS tables
sra.2019 <- sra.2019[ , c("Probe", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2019) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
# merge
sra.2019.ls <- lapply(split(sra.2019.df, sra.2019.df$lyr_name), function(df) {
  df <- merge(df, sra.2019[grep(df$lyr_name, sra.2019$ID), ])
  df$ID <- NULL
  df$PMeco <- paste0(df$PM, df$ECO)
  return(df)
})

# reshape list by PMeco
sra.2019.ls <- split(bind_rows(sra.2019.ls), bind_rows(sra.2019.ls)[["PMeco"]])
### Extract 14C data for incubation samples
## respired CO2, soil
# 2019
ams_results_ls_co2_S19 <- ams_results_ls[grep("co2-S19", names(ams_results_ls))]
for (i in seq_along(sra.2019.inc)) {
  sra.2019.inc[[i]] <- lapply(ams_results_ls_co2_S19, function(ls) {
    lapply(ls, function(df) {
      if (any(grepl(names(sra.2019.inc)[i], df$Probe))) {
        df[grep(names(sra.2019.inc)[i], df$Probe), ] 
      }
    })
  })
  sra.2019.inc[[i]] <- Filter(Negate(is.null), unlist(sra.2019.inc[[i]], recursive = FALSE))
}
sra.2019.inc <- type.convert(
  bind_rows(
    lapply(unlist(sra.2019.inc, recursive = FALSE), 
           function(x) x %>% mutate_all(as.character))),
  as.is = TRUE)
sra.2019.inc <- sra.2019.inc[-which(is.na(sra.2019.inc$F14C)), ]

# 2001
ams_results_ls_co2_S01 <- ams_results_ls[grep("co2-S01", names(ams_results_ls))]
# remove questionable/duplicate samples
# ANrf_comp_11-32_2001_a (analyzed twice; both anomously low compared to rep and rest of data)
ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_22.xlsx` <- ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_22.xlsx`[-grep("ANrf_comp_11-32_2001_a", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_22.xlsx`$Probe), ]
ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-16`$`14C_SierraNevada_Inc_2001_3_results.xlsx` <- ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-16`$`14C_SierraNevada_Inc_2001_3_results.xlsx`[-grep("ANrf_comp_11-32_2001_a", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-16`$`14C_SierraNevada_Inc_2001_3_results.xlsx`$Probe), ]
# from original analysis of samples extracted online 11-Dec-2020 (see readme for notes)
ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_23.xlsx` <- ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_23.xlsx`[-grep("GRwf_comp_13-28_2001_a_11", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx`$Probe), ]
# from reanalysis of samples extracted online 11-Dec-2020 (see readme for notes)
# GRrf_comp_8-27_2001_a_5, GRrf_comp_8-27_2001_b_6, GRpp_comp_15-27_2001_b_18 
ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx` <- ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx`[c(
  grep("GRrf_comp_8-27_2001_a_5", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx`$Probe),
  grep("GRwf_comp_13-28_2001_b_12", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx`$Probe)), ]

# create template for extracting data
sra.2001.inc <- vector(mode = "list", length = length(unique(sra.2019.inc.df$pro_name)))
names(sra.2001.inc) <- unique(sra.2019.inc.df$pro_name)
# merge with 14C data
for (i in seq_along(sra.2001.inc)) {
  sra.2001.inc[[i]] <- lapply(ams_results_ls_co2_S01, function(ls) {
    lapply(ls, function(df) {
      if (any(grepl(names(sra.2001.inc)[i], df$Probe))) {
        df[grep(names(sra.2001.inc)[i], df$Probe), ] 
      }
    })
  })
  sra.2001.inc[[i]] <- Filter(Negate(is.null), unlist(sra.2001.inc[[i]], recursive = FALSE))
}
sra.2001.inc <- type.convert(
  bind_rows(
    lapply(unlist(sra.2001.inc, recursive = FALSE), 
           function(x) x %>% mutate_all(as.character))),
  as.is = TRUE)
sra.2001.inc <- sra.2001.inc[-which(is.na(sra.2001.inc$F14C)), ]

# respired CO2, litter
ams_results_ls_co2_L19 <- ams_results_ls[grep("co2-L19", names(ams_results_ls))]
for(i in seq_along(sra.2019.inc_L)) {
  sra.2019.inc_L[[i]] <- lapply(ams_results_ls_co2_L19, function(ls) {
    lapply(ls, function(df) {
      if(any(grepl(names(sra.2019.inc_L)[i], df$Probe))) {
       df[grep(names(sra.2019.inc_L)[i], df$Probe), ] 
      }
    })
  })
  sra.2019.inc_L[[i]] <- Filter(Negate(is.null), unlist(sra.2019.inc_L[[i]], recursive = FALSE))
}
sra.2019.inc_L <- bind_rows(unlist(sra.2019.inc_L, recursive = FALSE))

## merge w/ templates [why do I do this twice?]
# rename cols in AMS tables
# soil CO2
sra.2019.inc <- sra.2019.inc[ , c("Probe", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2019.inc) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
sra.2001.inc <- sra.2001.inc[ , c("Probe", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2001.inc) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
# merge
# 2019
sra.2019.inc.ls <- bind_rows(
  lapply(split(sra.2019.inc.df, sra.2019.inc.df$lyr_name), function(df) {
    df <- merge(df, sra.2019.inc[grep(df$lyr_name, sra.2019.inc$ID), ])
    df$ID <- NULL
    df$PMeco <- paste0(df$PM, df$ECO)
    return(df)
  })
)
sra.2019.inc.ls <- split(sra.2019.inc.ls, sra.2019.inc.ls$PMeco)
# 2001
sra.2001.inc.ls <- bind_rows(
  lapply(split(sra.2001.inc.df, sra.2001.inc.df$lyr_name), function(df) {
    df <- merge(df, sra.2001.inc[grep(df$lyr_name, sra.2001.inc$ID), ])
    df$ID <- NULL
    df$PMeco <- paste0(df$PM, df$ECO)
    return(df)
  })
)
sra.2001.inc.ls <- split(sra.2001.inc.ls, sra.2001.inc.ls$PMeco)

# save inc list
save(sra.2001.inc.ls, file = "sra.2001.inc.ls.RData")

# litter CO2
sra.2019.inc_L <- sra.2019.inc_L[ , c("Probe", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2019.inc_L) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
sra.2019.inc_L$ID <- substr(substring(sra.2019.inc_L$ID, 
                                      regexpr("_", sra.2019.inc_L$ID) + 1, 
                                      nchar(sra.2019.inc_L$ID)),
                            1, 8)
sra.2019.inc.df_L <- data.frame(Year = rep(2019, 18),
                                rep = rep(c(1, 2), 9),
                                PM = rep(c("AN", "BS", "GR"), each = 6),
                                eco = rep(c("pp", "wf", "rf"), each = 2, times = 3))
sra.2019.inc.df_L$PMeco <- paste0(sra.2019.inc.df_L$PM, sra.2019.inc.df_L$eco)
sra.2019.inc.df_L$ID <- paste0(sra.2019.inc.df_L$PM, sra.2019.inc.df_L$eco, "-L_", sra.2019.inc.df_L$rep)
# add dry wts and litter depth
sra.2019.L <- read.csv("../data/derived/lab_jena_litter/Litter_2019_2021-01-27.csv")
sra.2019.inc.df_L <- merge(sra.2019.inc.df_L, sra.2019.L[ , c("PMeco", "lyr_top", "lyr_bot")], all.x = TRUE)
# merge
sra.2019.inc_L.df <- bind_rows(
  lapply(split(sra.2019.inc_L, sra.2019.inc_L$ID), function(df) {
    df <- merge(df, sra.2019.inc.df_L, by = "ID")
    df$ID <- NULL
    return(df)
  })
)
sra.2019.inc_L.ls <- split(sra.2019.inc_L.df, sra.2019.inc_L.df$PMeco)
# fm and d14c conversion functions
lambda <- 1/8267 # = 1/(true mean life of 14C)
calc_fm <- function(d14c, obs_date_y) {
  ((d14c/1000) + 1)/exp(lambda * (1950 - obs_date_y))
}
calc_14c <- function(fm, obs_date_y) {
  (fm * exp(lambda * (1950 - obs_date_y)) - 1) * 1000
}

# calc atm in 2001, 2009, 2019
Datm <- rbind(graven, future14C)
atm.d14.2001 <- Datm[Datm$Date == 2001.5, "NHc14"]
atm.fm.2001 <- calc_fm(atm.d14.2001, 2001)
atm.d14.2009 <- Datm[Datm$Date == 2009.5, "NHc14"]
atm.fm.2009 <- calc_fm(atm.d14.2009, 2009)
atm.d14.2019 <- Datm[Datm$Date == 2019.5, "NHc14"]
atm.fm.2019 <- calc_fm(atm.d14.2019, 2019)
fig.n <- fig.n + 1
# summarize litter inc data
sra.2019.inc_L.sum <- sra.2019.inc_L.df %>%
  mutate(eco = factor(ifelse(eco == "pp", "warm",
                             ifelse(eco == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = factor(ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))),
         Year = factor("2019")) %>%
  group_by(Year, PMeco, pm, eco, lyr_top, lyr_bot) %>%
  summarize(d14c_mean = mean(d14c),
            d14c_u = max(d14c),
            d14c_l = min(d14c),
            fm_mean = mean(fm),
            fm_u = max(fm),
            fm_l = min(fm))

# plot as cols by climate
sra.2019.inc_L.sum %>%
  mutate(MAT = factor(eco, levels = c("warm", "cool", "cold"), labels = c("10-13", "8-10", "5-6"))) %>%
  ggplot(., aes(MAT, d14c_mean, fill = pm)) +
  geom_hline(yintercept = 0) +
  geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
  geom_col(position = "dodge2") +
  geom_errorbar(aes(ymax = d14c_u, ymin = d14c_l, color = pm), 
                position = position_dodge2(width = .5, padding = .5)) +
  scale_fill_manual(name = "Parent material",
                    values = c("andesite" = "blue", 
                               "basalt" = "red", 
                               "granite" = "darkgray")) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  coord_flip() +
  ylab(expression(Delta*''^14*'C (‰)')) +
  xlab(expression("MAT ("*~degree*C*")")) +
  theme_bw() +
  theme(panel.grid = element_blank())


# plot as points with depth
sra.2019.inc_L.sum %>%
  ggplot(., aes(d14c_mean, lyr_top, color = pm)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_point(size = 3) +
  geom_errorbarh(aes(xmax = d14c_u, xmin = d14c_l), height = 1) +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  facet_grid(rows = vars(eco)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid = element_blank())

Fig. 16. Litter incubation \(\Delta\)14C-CO2 (2019)

Caption: Mean \(\Delta\)14C-CO2 for each site. Error bars show min and max of duplicate incubation samples. a) Data shown by site, without litter depth, b) Data shown by depth of litter layer, binned by climate zone.

pro.plot <- function(df, maxDepth, min14C, rep) {
  ggplot(df, aes(d14c, lyr_bot, color = PM, shape = ECO, group = rep)) +
    geom_vline(xintercept = 0) +
    geom_hline(yintercept = 0) +
    geom_point(size = 3) +
    geom_path() +
    scale_y_reverse(limits = c(maxDepth, 0)) +
    scale_x_continuous(limits = c(min14C, 180)) +
    scale_color_manual(name = "parent material",
                       labels = c("AN" = "andesite",
                                  "BS" = "basalt",
                                  "GR" = "granite"),
                       values = c("AN" = "blue", 
                                  "BS" = "red", 
                                  "GR" = "darkgray")) +
    scale_shape_manual(name = "ecosystem",
                       labels = c("pp" = expression(italic("P. ponderosa")),
                                  "rf" = expression(italic("A. magnifica")),
                                  "wf" = expression(italic("A. concolor"))),
                       values = c("pp" = 15, 
                                  "rf" = 16, 
                                  "wf" = 17)) +
    xlab(expression(Delta*''^14*'C (‰)')) +
    ylab("Depth (cm)") +
    theme_bw() +
    theme(panel.grid.minor = element_blank())
}
# lapply(sra.2001.ls, function(df) pro.plot(df, max(df$lyr_bot), min(df$d14c), df$pro_rep))
# lapply(sra.2019.ls, function(df) pro.plot(df, max(df$lyr_bot), min(df$d14c), df$pro_rep))
# lapply(sra.2019.inc.ls, function(df) pro.plot(df, max(df$lyr_bot), min(df$d14c), NA))

2001 mean radiocarbon profiles

# combine reps
sra.2001.sum.ls  <- lapply(sra.2001.ls, function(df) {
  df <- data.frame(df %>%
                     filter(lyr_bot <= 40) %>%
                     mutate(lyr_top_ch = as.character(lyr_top),
                            lyr_bot_ch = as.character(lyr_bot)) %>%
                     select(PM, ECO, PMeco, fm, d14c, lyr_top_ch, lyr_bot_ch) %>%
                     group_by(PM, ECO, PMeco, lyr_top_ch, lyr_bot_ch) %>%
                     summarize_all(list(mean = mean, sd = sd), na.rm = TRUE))
  names(df) <- c("PM", "ECO", "PMeco", "lyr_top", "lyr_bot", "fm", "d14c", "fm_sd", "d14c_sd")
  df$lyr_top <- as.numeric(df$lyr_top)
  df$lyr_bot <- as.numeric(df$lyr_bot)
  df$d14c_u <- df$d14c + df$d14c_sd
  df$d14c_l <- df$d14c - df$d14c_sd
  return(df[order(df$lyr_bot), ])
})
sra.01.sum <- bind_rows(sra.2001.sum.ls) %>%
  mutate(eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite")))

# plot
fig.n <- fig.n + 1
sra.01.sum %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, group = PMeco)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path() +
  scale_y_reverse(limits = c(40, 0)) +
  scale_x_continuous(limits = c(-100, 180)) +    
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. Mean profile \(\Delta\)14C for 2001 samples

Caption: Mean \(\Delta\)14C by depth for each site in 2001. Error bars show ±1 standard deviation, solid vertical line shows \(\Delta\)14C of the atmosphere in the year of sampling.

2009 radiocarbon profiles

# 2009 summary data (from C. Rasmussen)
ras18.sum <- read_excel(
  "../data/external/sra_ras_sum/sierra_data_summary_2020.xlsx",
  sheet = "Data_Summary_2018_paper")

# remove empty data rows
ras18.sum <- ras18.sum[-which(is.na(ras18.sum$`Sample ID`)), ]

# summarize 09 data
sra.09.sum <- ras18.sum %>%
  mutate(
    ECO = ifelse(Biome == "PP", "pp", ifelse(Biome == "WF", "wf", "rf")),
    PM = ifelse(Parent_Material == "Andesite", "AN", ifelse(Parent_Material == "Basalt", "BS", "GR")),
    eco = factor(ifelse(ECO == "pp", "warm", ifelse(ECO == "wf", "cool", "cold")),
                 levels = c("warm", "cool", "cold")),
    pm = paste0(tolower(substr(Parent_Material, 1, 1)), 
                substr(Parent_Material, 2, nchar(Parent_Material))),
    mass_kgm2 = BD_g_cm_3 * Soil_finefraction * (`bottom mineral` - `top mineral`) * 10) %>%
  mutate(PMeco = paste0(PM, ECO)) %>%
  rename(d14c = "Δ14C",
         lyr_bot = "bottom mineral",
         lyr_top = "top mineral")
sra.2009.ls <- lapply(split(sra.09.sum, sra.09.sum$PMeco), function(df) {
  df$lyr_fraction_modern <- calc_fm(df$d14c, 2009)
  return(data.frame(df))
})

# 2009 bulk C data
ras18.blkC <- as.data.frame(read_excel(
  "../data/external/sra_ras_sum/sierra_data_summary_2020.xlsx",
  sheet = "2009_bulk_data"))

# Add PM, ECO, mass_kgm2 vars
ras18.blkC$ECO <- ifelse(ras18.blkC$Biome == "PP", "pp", ifelse(ras18.blkC$Biome == "RF", "rf", "wf"))
ras18.blkC$PMeco <- paste0(ras18.blkC$PM, ras18.blkC$ECO)
ras18.blkC$mass_kgm2 <- ras18.blkC$Thickness_cm * ras18.blkC$BD_g_cm_3 * ras18.blkC$Soil_finefraction * 10

# Calculate SOC stocks
ras18.blkC$lyr_soc <- ras18.blkC$Thickness_cm * ras18.blkC$BD_g_cm_3 * ras18.blkC$Soil_finefraction * ras18.blkC$C_pct * 10^-1

# Calculate cmtv SOC stocks
ras18.blkC$lyr_soc_cmtv <- unlist(lapply(split(ras18.blkC, ras18.blkC$pro_name), function(x) {
  x$lyr_soc_cmtv <- NA
  for (i in seq(nrow(x))) {
    if (i == 1) {
      x$lyr_soc_cmtv[i] <- x$lyr_soc[i]
    } else {
      x$lyr_soc_cmtv[i] <- x$lyr_soc[i] + x$lyr_soc_cmtv[i - 1]
    }
  }
  return(x$lyr_soc_cmtv)
}))
# LOOCV function, fit = lm mod
loocv <- function (fit) {
  h <- lm.influence(fit)$h
  mean((residuals(fit) / (1-h))^2)
}

# test function for predicting BD as function of PM, ECO, and C content
bd.mod <- lm(BD_g_cm_3 ~ PM * ECO + PM * C_pct + `bottom mineral`, ras18.blkC)
bd.pred <- predict.lm(bd.mod, ras18.blkC, interval = "predict", pred.var = loocv(bd.mod))
bd.err.df <- ras18.blkC
bd.err.df$BD_pred <- bd.pred[ , 1]
bd.err.df$BD_l <- bd.pred[ , 2]
bd.err.df$BD_u <- bd.pred[ , 3]

# plot
ggplot(bd.err.df, aes(BD_g_cm_3, BD_pred)) +
  geom_ribbon(aes(ymin = BD_l, ymax = BD_u), fill = "lightgray") +
  geom_smooth(method = "lm", formula = y ~ x, se = FALSE, color = "black") +
  geom_point(aes(color = PM, shape = ECO, size = `bottom mineral`/10)) +
  scale_color_manual(name = "Parent material",
                     values = c("AN" = andesite,
                                "BS" = basalt,
                                "GR" = granite)) +
  theme_bw() +
  theme(panel.grid = element_blank())

# plot
fig.n <- fig.n + 1
sra.09.sum %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, group = PMeco)) +
  geom_vline(xintercept = atm.d14.2009) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_path(linetype = "dashed") +
  scale_y_reverse() +
  scale_x_continuous(limits = c(-100, 180)) +    
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "ecosystem",
                     values = c("warm" = 0, 
                                "cool" = 1, 
                                "cold" = 2)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. Profile \(\Delta\)14C for 2009 samples

Caption: Profile \(\Delta\)14C by depth for each site in 2009. Solid vertical line shows \(\Delta\)14C of the atmosphere in the year of sampling. Error bars not shown as only a single replicate profile was analyzed per site.

2019 mean radiocarbon profiles

# combine reps
sra.2019.sum.ls  <- lapply(sra.2019.ls, function(df) {
  df <- data.frame(df %>%
                     mutate(lyr_top_ch = as.character(lyr_top),
                            lyr_bot_ch = as.character(lyr_bot)) %>%
                     select(PM, ECO, PMeco, fm, d14c, lyr_top_ch, lyr_bot_ch) %>%
                     group_by(PM, ECO, PMeco, lyr_top_ch, lyr_bot_ch) %>%
                     summarize_all(list(mean = mean, sd = sd), na.rm = TRUE))
  names(df) <- c("PM", "ECO", "PMeco", "lyr_top", "lyr_bot", "fm", "d14c", "fm_sd", "d14c_sd")
  df$lyr_top <- as.numeric(df$lyr_top)
  df$lyr_bot <- as.numeric(df$lyr_bot)
  df$d14c_u <- df$d14c + df$d14c_sd
  df$d14c_l <- df$d14c - df$d14c_sd
  return(df)
})
sra.19.sum <- bind_rows(sra.2019.sum.ls) %>%
  mutate(eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) 

# plot
fig.n <- fig.n + 1
sra.19.sum %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, group = PMeco)) +
  geom_vline(xintercept = atm.d14.2019) +
  geom_hline(yintercept = 0) +
  geom_point(size = 2.7) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path() +
  scale_y_reverse(limits = c(max(sra.19.sum$lyr_bot), 0)) +
  scale_x_continuous(limits = c(min(sra.19.sum$d14c), 180)) +    
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. Mean profile \(\Delta\)14C for 2019 samples

Caption: Mean \(\Delta\)14C by depth for each site in 2019. Error bars show ±1 standard deviation, solid vertical line shows \(\Delta\)14C of the atmosphere in the year of sampling.

Change in \(\Delta\)14C over time between 2001 and 2019

# combine '01 and '19 data for plotting
sra.01.sum$Year <- 2001
sra.19.sum$Year <- 2019

sra.01.19.sum <- rbind(sra.01.sum, sra.19.sum)
sra.01.19.sum$Year <- as.factor(sra.01.19.sum$Year)

fig.n <- fig.n + 1
sra.01.19.sum %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         ecoYear = paste0(ECO, Year),
         ecoYear2 = ifelse(ecoYear == "pp2001", "warm (2001)",
                           ifelse(ecoYear == "pp2019", "warm (2019)",
                                  ifelse(ecoYear == "wf2001", "cool (2001)",
                                         ifelse(ecoYear == "wf2019", "cool (2019)",
                                                ifelse(ecoYear == "rf2001", "cold (2001)", "cold (2019)")))))) %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = ecoYear2, group = PMeco_year)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path(aes(linetype = Year)) +
  scale_y_reverse(limits = c(41, 0)) +
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. Mean profile \(\Delta\)14C for 2001 and 2019 samples

Caption: Mean \(\Delta\)14C by depth for each site in 2001 and 2019. Error bars show ±1 standard deviation. Vertical lines show \(\Delta\)14C of the atmosphere in 2001 (solid) and 2019 (dashed).

Incubation \(\Delta\)14C-CO2

## 2019
sra.2019.inc.df <- bind_rows(sra.2019.inc.ls)
# add litter inc data and summarize
sra.2019.inc.sum.df <- data.frame(rbind(
  sra.2019.inc_L.sum,
  sra.2019.inc.df %>%
    mutate(eco = factor(ifelse(ECO == "pp", "warm",
                             ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
           pm = factor(ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))),
           # remove GRrf 10-20 "a" point
           d14c = replace(d14c, which(d14c < -300), NA),
           Year = factor(Year)) %>%
  group_by(Year, PMeco, pm, eco, lyr_bot, lyr_top) %>%
  summarize(d14c_mean = mean(d14c, na.rm = TRUE),
            d14c_l = min(d14c, na.rm = TRUE),
            d14c_u = max(d14c, na.rm = TRUE),
            fm_mean = mean(fm),
            fm_l = min(fm),
            fm_u = max(fm))))
sra.2019.inc.sum.ls <- split(sra.2019.inc.sum.df, sra.2019.inc.sum.df$PMeco)

# 2001
sra.2001.inc.df <- bind_rows(sra.2001.inc.ls)
sra.2001.inc.sum.df <- data.frame(
  sra.2001.inc.df %>%
    mutate(eco = factor(ifelse(ECO == "pp", "warm",
                               ifelse(ECO == "wf", "cool", "cold")),
                        levels = c("warm", "cool", "cold")),
           pm = factor(ifelse(PM == "AN", "andesite",
                       ifelse(PM == "BS", "basalt", "granite"))),
           Year = factor(Year)) %>%
    group_by(Year, PMeco, pm, eco, lyr_bot, lyr_top) %>%
    summarize(d14c_mean = mean(d14c),
              d14c_l = min(d14c),
              d14c_u = max(d14c),
              fm_mean = mean(fm),
              fm_l = min(fm),
              fm_u = max(fm))
)
sra.2001.inc.sum.ls <- split(sra.2001.inc.sum.df, sra.2001.inc.sum.df$PMeco)
sra.2001.inc.sum.df <- sra.2001.inc.sum.df[ , !(names(sra.2001.inc.sum.df) %in% c("fm_mean", "fm_l", "fm_u", "lyr_top", "PMeco"))]
# 2019
fig.n <- fig.n + 1
sra.2019.inc.sum.df[order(sra.2019.inc.sum.df$lyr_bot), ] %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = eco)) +
  geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l,
        xmax = d14c_u,
        color = pm),
    height = 1.5) +
  geom_path(linetype = "dashed") +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 0, 
                                "cool" = 1, 
                                "cold" = 2)) +
  xlab(expression('Incubation '*Delta*''^14*'C-CO'[2]*' (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. \(\Delta\)14C-CO2 of 2019 bulk soil incubations

Caption: \(\Delta\)14CO2 by depth for each site in 2019. One rep from GRrf 10-20 (the 10-20 cm increment sample from the cold granite site) is strongly depleted relative to the other rep: \(\Delta\)14C-CO2 = -210.1. The highly depleted sample has been excluded for display reasons.

# plot 2001 data
fig.n <- fig.n + 1
sra.2001.inc.sum.df %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = eco)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l,
        xmax = d14c_u,
        color = pm),
    height = 1.5) +
  geom_path() +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue",
                                "basalt" = "red",
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15,
                                "cool" = 16,
                                "cold" = 17)) +
  scale_x_continuous(limits = c(-70, 190)) +
  xlab(expression('Incubation '*Delta*''^14*'C-CO'[2]*' (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. \(\Delta\)14C-CO2 of 2001 bulk soil incubations

Caption: \(\Delta\)14CO2 by depth for each site in 2001. Note that some sites only have two depth increments. Similar to the 2019 dataset, one of the GRrf reps from the deepest depth increment was strongly depleted: \(\Delta\)14C-CO2 = -469.1. Both points have been excluded for display reasons.

# plot together
sra.inc.all <- rbind(sra.2001.inc.sum.df, 
                     sra.2019.inc.sum.df[ , names(sra.2019.inc.sum.df) %in% names(sra.2001.inc.sum.df)])
save(sra.inc.all, file = "sra.inc.all.RData")

fig.n <- fig.n + 1 
sra.inc.all %>%
  filter(lyr_bot > 0) %>%
  mutate(PMeco_year = paste0(pm, eco, Year),
         ecoYear = paste0(eco, " (", Year, ")")) %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = ecoYear, group = PMeco_year)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path(aes(linetype = Year)) +
  scale_y_reverse(limits = c(41, 0)) +
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_x_continuous(limits = c(-70, 190)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. \(\Delta\)14C-CO2 of 2001 and 2019 bulk soil incubations

Caption: \(\Delta\)14CO2 by depth for each site in 2001 and 2019. Different depth increments were sampled in 2001 and 2019. Points are the mean of laboratory duplicates; error bars are the measured values of each duplicate. Granite/cold point exlcuded for display reasons as it is strongly depleted.

Incubation vs. bulk soil \(\Delta\)14C

# bind rows of inc list
sra.19.inc <- sra.2019.inc.sum.df
sra.19.inc$Type <- "inc"

# 2001
sra.01.inc <- sra.2001.inc.sum.df
sra.01.inc$Type <- "inc"

# rbind bulk data
sra.19.bulk <- sra.19.sum[which(sra.19.sum$lyr_bot < 31), c("Year", "PM", "ECO", "lyr_bot","d14c", "d14c_l", "d14c_u")]
names(sra.19.bulk)[which(names(sra.19.bulk) == "d14c")] <- "d14c_mean"
sra.19.bulk$Type <- "bulk"
sra.19.bulk <- sra.19.bulk %>%
  mutate(pm = factor(ifelse(PM == "AN", "andesite",
                            ifelse(PM == "BS", "basalt", "granite"))),
         eco = factor(ifelse(ECO == "pp", "warm",
                             ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         PM = NULL,
         ECO = NULL)
sra.19.inc.blk <- rbind(sra.19.bulk, sra.19.inc[ , names(sra.19.inc) %in% names(sra.19.bulk)])
save(sra.19.inc.blk, file = "sra.19.inc.blk.RData")

# 2001
# Need to calculate weighted average of radiocarbon values and stocks for combined inc depths 
# 1) add SOC stocks to duplicate sra.2001.ls obj
sra.2001.ls2 <- sra.2001.ls
for(i in seq_along(sra.2001.ls2)) {
  ix <- match(sra.2001.ls2[[i]][["ID"]], soc.2001.ls[[i]][["ID"]])
  sra.2001.ls2[[i]]["lyr_soc_kgm2"] <- soc.2001.ls[[i]][ix, "lyr_soc_kgm2"]
}
# 2) weighted average fx
d1d2.14c.fx <- function(df) {
  sum_soc <- sum(df[1:2, "lyr_soc_kgm2"])
  wt1 <- df$lyr_soc_kgm2[1] / sum_soc
  wt2 <- df$lyr_soc_kgm2[2] / sum_soc
  d1d2 <- df[1, ]
  d1d2$ID = paste(df$PMeco[1], df$pro_rep[1], df$lyr_top[1], df$lyr_bot[2], sep = "_")
  d1d2$lyr_soc_kgm2 = sum(df$lyr_soc_kgm2[1], df$lyr_soc_kgm2[2])
  d1d2$lyr_bot = df$lyr_bot[2]
  d1d2$fm <- sum(df$fm[1] * wt1, df$fm[2] * wt2)
  d1d2$d14c <- sum(df$d14c[1] * wt1, df$d14c[2] * wt2)
  return(rbind(d1d2,
               df[3:nrow(df), ]))
}
# 3) calc. wtd. average for GRrf
sra.2001.ls2$GRrf <- bind_rows(
  lapply(split(sra.2001.ls2$GRrf, sra.2001.ls2$GRrf$pro_rep), function(x) {
    d1d2.14c.fx(x)
  })
)
# 4) calc. wtd. average for BSrf
#    - problem here is that only one pro_rep has 0-3 cm data
#    - so, need to calculate weighted SOC, then calculate weighted 14C
#    - composite 0-8 = 15g BSrf_1_0-3 + 5 g from each pro_rep BSrf_3-8
BSrf_comp_01_i <- sra.2001.ls2$BSrf[which(sra.2001.ls$BSrf$lyr_bot < 9), ]
BSrf_comp_01_i$soc_wt <- c(15 / 30, rep(5 / 30, 3))
BSrf_comp_01_i$soc_wtd <- BSrf_comp_01_i$lyr_soc_kgm2 * BSrf_comp_01_i$soc_wt

# create summarized list
sra.2001.sum.ls2  <- lapply(sra.2001.ls2, function(df) {
  data.frame(
    df %>%
      filter(lyr_bot <= 40) %>%
      mutate(lyr_bot_ch = as.character(lyr_bot)) %>%
      select(PMeco, d14c, fm, lyr_bot_ch, lyr_soc_kgm2) %>%
      group_by(PMeco, lyr_bot_ch) %>%
      summarize(
        across(where(is.numeric), list(mean = mean, sd = sd), na.rm = TRUE)) %>%
      mutate(lyr_bot = as.numeric(lyr_bot_ch)) %>%
      select(-lyr_bot_ch)
  )
})

# remove BSrf row w/ lyr_bot = 3
sra.2001.sum.ls2$BSrf <- sra.2001.sum.ls2$BSrf[-which(sra.2001.sum.ls2$BSrf$lyr_bot == 3), ]
# calculate weighted average for d14c, fm, lyr_soc_kgm2
sra.2001.sum.ls2$BSrf[which(sra.2001.sum.ls2$BSrf$lyr_bot == 8), "d14c_mean"] <- sum(BSrf_comp_01_i$d14c * (BSrf_comp_01_i$soc_wtd / sum(BSrf_comp_01_i$soc_wtd)))
sra.2001.sum.ls2$BSrf[which(sra.2001.sum.ls2$BSrf$lyr_bot == 8), "fm_mean"] <- sum(BSrf_comp_01_i$fm * (BSrf_comp_01_i$soc_wtd / sum(BSrf_comp_01_i$soc_wtd)))
sra.2001.sum.ls2$BSrf[which(sra.2001.sum.ls2$BSrf$lyr_bot == 8), "lyr_soc_kgm2_mean"] <- sum(BSrf_comp_01_i$soc_wtd)
sra.2001.sum.ls2$BSrf[which(sra.2001.sum.ls2$BSrf$lyr_bot == 8), c("d14c_sd", "fm_sd", "lyr_soc_kgm2_sd")] <- NA

# calculate cmtv soc
sra.2001.sum.ls2 <- lapply(sra.2001.sum.ls2, function(x) {
  x <- x[order(x$lyr_bot), ]
  x$lyr_soc_cmtv <- NA
  for(i in seq_along(x$lyr_bot)) {
      if(i == 1) {
        x$lyr_soc_cmtv[i] <- x$lyr_soc_kgm2_mean[i]
      } else {
        x$lyr_soc_cmtv[i] <- x$lyr_soc_kgm2_mean[i] + x$lyr_soc_cmtv[i-1] 
      }
  }
  return(x)
})

# make df
sra.01.sum <- data.frame(bind_rows(
  lapply(sra.2001.sum.ls2, function(df) {
    df %>%
      mutate(eco = factor(ifelse(grepl("pp", df$PMeco), "warm",
                                 ifelse(grepl("wf", df$PMeco), "cool", "cold")),
                          levels = c("warm", "cool", "cold")),
             pm = ifelse(grepl("AN", df$PMeco), "andesite",
                         ifelse(grepl("BS", df$PMeco), "basalt", "granite")),
             d14c_u = d14c_mean + d14c_sd,
             d14c_l = d14c_mean - d14c_sd,
             Year = 2001,
             Type = "bulk") %>%
      select(names(sra.01.inc)) %>%
      arrange(lyr_bot)
  })
))
# bind with inc
sra.01.inc.blk <- rbind(data.frame(sra.01.inc), sra.01.sum)
save(sra.01.inc.blk, file = "sra.01.inc.blk.RData")
ts <- bind_rows(sra.19a.co2.ts[ , nms], 
                sra.19b.co2.ts[ , nms], 
                sra.01.1.co2.ts[ , nms],
                sra.01.2.co2.ts[ , nms])
if(length(which(is.na(ts$mgCO2_jar))) > 0) {
  ts <- ts[-which(is.na(ts$mgCO2_jar)), ]
}
ts$year <- sapply(strsplit(as.character(ts$ID), "_"), "[[", 3)
ts$rep <- sapply(strsplit(as.character(ts$ID), "_"), "[[", 4)
ts$depth <- sapply(strsplit(as.character(ts$ID), "_"), "[[", 2)
ts$ID2 <- paste(ts$PMeco, ts$depth, sep = "_")

# add C content
ts[which(ts$year == 2001), "gC_gS"] <- soc.2001.sum2[match(ts[which(ts$year == 2001), "ID2"], soc.2001.sum2$ID2), "c_pct_avg"] * 10^-2
Error in soc.2001.sum2[match(ts[which(ts$year == 2001), "ID2"], soc.2001.sum2$ID2),  : 
  non-numeric argument to binary operator
fig.n <- 1
# function for plotting
ts.plot.fx <- function(df, yr, increment, cumulative = TRUE) {
      if (cumulative) {
        df %>%
          filter(year == yr & depth_index == increment) %>%
          mutate(PM = ifelse(grepl("AN", PMeco), "AN",
                             ifelse(grepl("BS", PMeco), "BS", "GR")),
                 eco = factor(ifelse(grepl("rf", PMeco), "rf", 
                                     ifelse(grepl("wf", PMeco), "wf", "pp")),
                              levels = c("pp", "wf", "rf"))) %>%
          ggplot(., aes(time_d, mgCO2_gC_avg, color = PM, shape = eco)) +
          geom_ribbon(aes(ymin = mgCO2_gC_max, ymax = mgCO2_gC_min, fill = PM, linetype = eco, alpha = 0.2), show.legend = FALSE) +
          geom_point(aes(time_d, mgCO2_gC_max, fill = PM, shape = eco), color = "black", size = 3, stroke = 1) +
          geom_point(aes(time_d, mgCO2_gC_min, fill = PM, shape = eco), color = "black", size = 3, stroke = 1) +
          geom_line(aes(color = PM, linetype = eco), size = 1.2) +
          facet_grid(rows = vars(eco),
                     labeller = labeller(eco = c("rf" = "cold", "wf" = "cool", "pp" = "warm"))) +
          scale_x_continuous(limits = c(0,30)) +
          scale_color_manual(name = "Parent material",
                             labels = c("AN" = "andesite",
                                        "BS" = "basalt",
                                        "GR" = "granite"),
                             values = c("AN" = "blue", 
                                        "BS" = "red", 
                                        "GR" = "darkgray")) +
          scale_shape_manual(name = "Climate",
                             labels = c("rf" = "cold",
                                        "wf" = "cool",
                                        "pp" = "warm"),
                             values = c("pp" = 21, 
                                        "rf" = 22, 
                                        "wf" = 24)) +
          scale_fill_manual(values =c("AN" = "blue",
                                      "BS" = "red",
                                      "GR" = "darkgray")) +
          scale_linetype_manual(name = "Climate",
                                values = c("rf" = "dotted",
                                           "wf" = "dashed",
                                           "pp" = "solid"),
                                labels = c("rf" = "cold",
                                        "wf" = "cool",
                                        "pp" = "warm")) +
          ylab(expression('Cumulative flux (mgCO'[2]*'-C gC'^-1*')')) +
          xlab("Time (days)") +
          guides(color = guide_legend(order = 1),
                 shape = guide_legend(order = 3)) +
          ggtitle(paste("Cumulative flux, ", yr, "depth ", increment)) +
          theme_bw() +
          theme(panel.grid = element_blank())
    } else {
       df %>%
        filter(year == yr & depth_index == increment) %>%
        mutate(PM = ifelse(grepl("AN", PMeco), "AN",
                           ifelse(grepl("BS", PMeco), "BS", "GR")),
              eco = factor(ifelse(grepl("rf", PMeco), "rf",
                                  ifelse(grepl("wf", PMeco), "wf", "pp")),
                           levels = c("pp", "wf", "rf"))) %>%
        ggplot(., aes(time_d, mgCO2_gC_d_avg, color = PM, shape = eco)) +
        geom_ribbon(aes(ymin = mgCO2_gC_d_max, ymax = mgCO2_gC_d_min, fill = PM, linetype = eco, alpha = 0.2), show.legend = FALSE) +
          geom_point(aes(time_d, mgCO2_gC_d_max, fill = PM, shape = eco), color = "black", size = 3, stroke = 1) +
          geom_point(aes(time_d, mgCO2_gC_d_min, fill = PM, shape = eco), color = "black", size = 3, stroke = 1) +
        geom_line(aes(color = PM, linetype = eco), size = 1.2) +
        facet_grid(rows = vars(eco),
                   labeller = labeller(eco = c("rf" = "cold", "wf" = "cool", "pp" = "warm"))) +
        scale_x_continuous(limits = c(0,30)) +
        scale_color_manual(name = "Parent material",
                           labels = c("AN" = "andesite",
                                      "BS" = "basalt",
                                      "GR" = "granite"),
                           values = c("AN" = "blue", 
                                      "BS" = "red", 
                                      "GR" = "darkgray")) +
        scale_shape_manual(name = "Climate",
                           labels = c("rf" = "cold",
                                      "wf" = "cool",
                                      "pp" = "warm"),
                           values = c("pp" = 21, 
                                      "rf" = 22, 
                                      "wf" = 24)) +
        scale_fill_manual(values =c("AN" = "blue",
                                    "BS" = "red",
                                    "GR" = "darkgray")) +
        scale_linetype_manual(name = "Climate",
                              values = c("rf" = "dotted",
                                         "wf" = "dashed",
                                         "pp" = "solid"),
                              labels = c("rf" = "cold",
                                      "wf" = "cool",
                                      "pp" = "warm")) +
        ylab(expression('Respiration Rate (mgCO'[2]*'-C gC'^-1*'d'^-1*')')) +
        xlab("Time (days)") +
        guides(color = guide_legend(order = 1),
               shape = guide_legend(order = 3)) +
        ggtitle(paste("Flux rate", yr, "depth ", increment)) +
        theme_bw() +
        theme(panel.grid = element_blank())
    }
}

## cumulative flux
# 2019
ts.plot.fx(ts.avg, yr = "2019", increment = "1")
ts.plot.fx(ts.avg, yr = "2019", increment = "2")
ts.plot.fx(ts.avg, yr = "2019", increment = "3")
# 2001
ts.plot.fx(ts.avg, yr = "2001", increment = "1")
ts.plot.fx(ts.avg, yr = "2001", increment = "2")
ts.plot.fx(ts.avg, yr = "2001", increment = "3")

## flux rates
# 2019
ts.plot.fx(ts.avg, yr = "2019", increment = "1", cumulative = FALSE)
ts.plot.fx(ts.avg, yr = "2019", increment = "2", cumulative = FALSE)
ts.plot.fx(ts.avg, yr = "2019", increment = "3", cumulative = FALSE)
# 2001
ts.plot.fx(ts.avg, yr = "2001", increment = "1", cumulative = FALSE)
ts.plot.fx(ts.avg, yr = "2001", increment = "2", cumulative = FALSE)
ts.plot.fx(ts.avg, yr = "2001", increment = "3", cumulative = FALSE)

Fig. 16. Respiration data from incubations of 2019 and 2001 bulk soils.

Caption: Points show measured CO2 production of laboratory duplicates as cumulative fluxes or daily flux rates by depth, lines show the means, and the ribbon represents the range.

# plot 2019
fig.n <- fig.n + 1
# p <-
sra.19.inc.blk %>%
  mutate(ECOtype = paste0(eco, " (", Type, ")")) %>%
  arrange(lyr_bot) %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = ECOtype, linetype = Type)) +
  geom_vline(xintercept = atm.d14.2019) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path() +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem (type)",
                     values = c("warm (bulk)" = 15, 
                                "cool (bulk)" = 16, 
                                "cold (bulk)" = 17,
                                "warm (inc)" = 0, 
                                "cool (inc)" = 1, 
                                "cold (inc)" = 2)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# ggsave("sra.bulkInc.19.pdf", p, device = cairo_pdf, width = 9.5, height = 5, units = "in")

Fig. 16. \(\Delta\)14C of 2019 bulk soil incubations and corresponding bulk soil

Caption: \(\Delta\)14C of bulk soil and respired CO2 by depth for each site in 2019. Error bars show one standard deviation for bulk soil, points show mean of three replicate profiles for bulk soils and single observations for respired CO2.

# plot 2001
fig.n <- fig.n + 1
sra.01.inc.blk %>%
  mutate(ECOtype = paste0(eco, " (", Type, ")")) %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = ECOtype, linetype = Type)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path() +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem (type)",
                     values = c("warm (bulk)" = 15, 
                                "cool (bulk)" = 16, 
                                "cold (bulk)" = 17,
                                "warm (inc)" = 0, 
                                "cool (inc)" = 1, 
                                "cold (inc)" = 2)) +
  scale_x_continuous(limits = c(-100, 200)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. \(\Delta\)14C of 2001 bulk soil incubations and corresponding bulk soil

Caption: \(\Delta\)14C of bulk soil and respired CO2 by depth for each site in 2001. Points show mean of three replicate profiles for bulk soils and mean of laboratory duplicates for respired CO2. The incubated soil samples are a composite made by homogenizing subsamples from each of the three replicate profile samples by depth. Error bars show one standard deviation for bulk soil and the measured values from laboratory duplicates of the incubated composite samples.

# first merge mean 14C data from 2019 samples with composite incubation data
nms.inc.blk <- c("pm", "eco", "lyr_bot", "Year")
sra.19.inc.blk2 <- left_join(sra.19.bulk %>% mutate(., Year = as.factor(Year)),
                             sra.2019.inc.sum.df,
                             by = nms.inc.blk,
                             suffix = c(".bulk", ".inc"))
# 2001
sra.01.inc.blk2 <- left_join(sra.01.sum %>% mutate(., Year = as.factor(Year)),
                             sra.01.inc,
                             by = nms.inc.blk,
                             suffix = c(".bulk", ".inc"))
sra.01.inc.blk2$PMeco <- paste0(sra.01.inc.blk2$pm, sra.01.inc.blk2$eco)
# add depth factor
sra.01.inc.blk2 <- unsplit(
  lapply(split(sra.01.inc.blk2, sra.01.inc.blk2$PMeco), function(x) {
  x$depth <- seq(1, nrow(x))
  return(x) 
  }), 
sra.01.inc.blk2$PMeco)
sra.01.inc.blk2 <- sra.01.inc.blk2[which(sra.01.inc.blk2$lyr_bot < 35), ]
sra.01.inc.blk2$depth <- factor(sra.01.inc.blk2$depth)

# regress bulk vs. inc
min.inc.blk.19 <- min(sra.19.inc.blk2$d14c_l.inc,
                      sra.19.inc.blk2$d14c_l.bulk) # exclude highly negative incubation sample from GRwf
max.inc.blk.19 <- max(sra.19.inc.blk2$d14c_l.inc,
                      sra.19.inc.blk2$d14c_l.bulk)

# What is the ideal grouping/expected relationship?
## look at combinatorial dataset
# sra.all.df.fx <- function(ls, year) {
#   cbind(bind_rows(lapply(ls, function(df) df[ , c("PMeco", "lyr_bot", "d14c")])),
#         year = year)
# }
# sra.all.df <- inner_join(
#   rbind(sra.all.df.fx(sra.2001.ls, 2001),
#         sra.all.df.fx(sra.2019.ls, 2019)),
#   rbind(sra.all.df.fx(sra.2001.inc.ls, 2001),
#         sra.all.df.fx(sra.2019.inc.ls, 2019)),
#   by = c("PMeco", "lyr_bot", "year"),
#   suffix = c("_bulk", "_inc"))
# sra.all.df <- sra.all.df %>%
#   mutate(PM = substr(PMeco, 1, 2),
#          ECO = substr(PMeco, 3, 4))
# 
# sra.all.df %>%
#   filter(d14c_inc > -130) %>%
#   ggplot(., aes(d14c_bulk, d14c_inc, color = PM)) +
#   geom_vline(xintercept = 0) +
#   geom_hline(yintercept = 0) +
#   geom_abline(slope = 1, intercept = 0) +
#   geom_smooth(method = "lm", formula = y ~ x, aes(fill = PM)) +
#   geom_point() +
#   scale_color_manual(name = "Parent material",
#                      values = c("AN" = "blue",
#                                 "BS" = "red",
#                                 "GR" = "darkgray"),
#                      labels = c("AN" = "andesite",
#                                 "BS" = "basalt",
#                                 "GR" = "granite")) +
#     scale_fill_manual(name = "Parent material",
#                      values = c("AN" = "blue",
#                                 "BS" = "red",
#                                 "GR" = "darkgray"),
#                      labels = c("AN" = "andesite",
#                                 "BS" = "basalt",
#                                 "GR" = "granite")) +
#   coord_fixed(xlim = c(-130, 200), ylim = c(-130, 200)) +
#   xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
#   ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
#   
# summary(lm(d14c_inc ~ d14c_bulk * PM, sra.all.df[sra.all.df$d14c_inc > -130, ]))

# join all data as means and sds
sra.all.sum.df <- left_join(
  bind_rows(sra.2001.sum.ls2) %>%
    select(PMeco, lyr_bot, d14c_mean, d14c_sd) %>%
    mutate(Year = 2001) %>%
    bind_rows(., 
              bind_rows(lapply(sra.2019.ls, function(df) {
                df %>%
                  filter(lyr_bot < 31) %>%
                  select(PMeco, lyr_bot, d14c) %>%
                  group_by(PMeco, lyr_bot) %>%
                  summarize(across(d14c, list(mean = mean, sd = sd))) %>%
                  mutate(Year = 2019)
                }))
            ),
  bind_rows(lapply(sra.2001.inc.ls, function(df) {
              df %>%
                select(PMeco, lyr_bot, d14c) %>%
                group_by(PMeco, lyr_bot) %>%
                summarize(across(d14c, list(mean = mean, sd = sd))) %>%
                mutate(Year = 2001) 
              })) %>%
  bind_rows(., 
            bind_rows(lapply(sra.2019.inc.ls, function(df) {
              df %>%
                select(PMeco, lyr_bot, d14c) %>%
                group_by(PMeco, lyr_bot) %>%
                summarize(across(d14c, list(mean = mean, sd = sd))) %>%
                mutate(Year = 2019)
              }))
            ), 
  by = c("PMeco", "lyr_bot", "Year"),
  suffix = c(".bulk", ".inc")) %>%
  mutate(PM = substring(PMeco, 1, 2),
         eco = substring(PMeco, 3, 4))

# Trend for means
# NB Year only, depth only models do not show significant interactions
# PM only model
emtrends(lm(d14c_mean.inc ~ d14c_mean.bulk * PM, sra.all.sum.df[sra.all.sum.df$d14c_mean.inc > -200, ]), pairwise ~ PM, var = "d14c_mean.bulk")
$emtrends
 PM d14c_mean.bulk.trend    SE df lower.CL upper.CL
 AN                0.514 0.144 42    0.224    0.804
 BS                0.873 0.169 42    0.532    1.214
 GR                0.952 0.136 42    0.679    1.226

Confidence level used: 0.95 

$contrasts
 contrast estimate    SE df t.ratio p.value
 AN - BS   -0.3593 0.222 42 -1.621  0.2481 
 AN - GR   -0.4387 0.198 42 -2.221  0.0792 
 BS - GR   -0.0794 0.216 42 -0.367  0.9287 

P value adjustment: tukey method for comparing a family of 3 estimates 
# ECO only model
emtrends(lm(d14c_mean.inc ~ d14c_mean.bulk * eco, sra.all.sum.df[sra.all.sum.df$d14c_mean.inc > -200, ]), pairwise ~ eco, var = "d14c_mean.bulk")
$emtrends
 eco d14c_mean.bulk.trend    SE df lower.CL upper.CL
 pp                 0.779 0.145 42    0.487    1.071
 rf                 0.777 0.317 42    0.137    1.417
 wf                 0.606 0.147 42    0.308    0.903

Confidence level used: 0.95 

$contrasts
 contrast estimate    SE df t.ratio p.value
 pp - rf   0.00222 0.349 42 0.006   1.0000 
 pp - wf   0.17373 0.206 42 0.841   0.6797 
 rf - wf   0.17150 0.350 42 0.491   0.8762 

P value adjustment: tukey method for comparing a family of 3 estimates 
# lapply(split(sra.all.sum.df, sra.all.sum.df$eco), function(df) {
#   summary(lm(d14c_mean.inc ~ d14c_mean.bulk * PM, df))
# })

# # Deming regression (accounts for error in x and y terms)
# sra.dem <- lapply(split(sra.all.sum.df, sra.all.sum.df$PM), function(df) {
#   deming(d14c_mean.inc ~ d14c_mean.bulk,
#        data = df, xstd = d14c_sd.inc, ystd = d14c_sd.bulk)
# })

# all depths and years together, by PM
fig.n <- fig.n + 1
sra.19.inc.blk2  %>%
  bind_rows(., sra.01.inc.blk2[ , which(names(sra.19.inc.blk2) %in% names(sra.01.inc.blk2))]) %>%
  mutate(depth = factor(lyr_bot),
         ecoYear = paste0(eco, " (", Year, ")")) %>%
  filter(d14c_mean.inc > -200) %>%
  ggplot(., aes(d14c_mean.bulk, d14c_mean.inc, color = pm)) +
  # geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_vline(xintercept = 0) +
  # geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_abline(slope = 1, intercept = 0) +
  geom_point(aes(color = pm, shape = ecoYear), size = 3) +
  geom_smooth(method = "lm", formula = y ~ x, se = FALSE) +
  geom_errorbarh(
    aes(xmin = d14c_l.bulk, 
        xmax = d14c_u.bulk,
        color = pm), 
    height = 1.5) +
  geom_errorbar(
    aes(ymin = d14c_l.inc, 
        ymax = d14c_u.inc,
        color = pm), 
    width = 1.5) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Ecosystem (year)",
                     values = c("warm (2019)" = 0,
                                "cool (2019)" = 2,
                                "cold (2019)" = 1,
                                "warm (2001)" = 15,
                                "cool (2001)" = 17,
                                "cold (2001)" = 16)) +
  coord_fixed(xlim = c(-100, 200), ylim = c(-100, 200)) +
  xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
  ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
  # facet_grid(rows = vars(depth)) +
  theme_bw() +
  theme(panel.grid = element_blank())


sra.19.inc.blk2  %>%
  bind_rows(., sra.01.inc.blk2[ , which(names(sra.19.inc.blk2) %in% names(sra.01.inc.blk2))]) %>%
  mutate(depth = factor(lyr_bot),
         ecoYear = paste0(eco, " (", Year, ")")) %>%
  filter(d14c_mean.inc > -200) %>%
  ggplot(., aes(d14c_mean.bulk, d14c_mean.inc, color = eco)) +
  # geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_vline(xintercept = 0) +
  # geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_abline(slope = 1, intercept = 0) +
  geom_point(aes(color = eco, shape = ecoYear), size = 3) +
  geom_smooth(method = "lm", formula = y ~ x, se = FALSE) +
  geom_errorbarh(
    aes(xmin = d14c_l.bulk, 
        xmax = d14c_u.bulk,
        color = eco), 
    height = 1.5) +
  geom_errorbar(
    aes(ymin = d14c_l.inc, 
        ymax = d14c_u.inc,
        color = eco), 
    width = 1.5) +
  scale_color_manual(name = "Climate",
                     values = c("warm" = warm,
                                "cool" = cool,
                                "cold" = cold)) +
  scale_shape_manual(name = "Ecosystem (year)",
                     values = c("warm (2019)" = 0,
                                "cool (2019)" = 1,
                                "cold (2019)" = 2,
                                "warm (2001)" = 15,
                                "cool (2001)" = 16,
                                "cold (2001)" = 17)) +
  coord_fixed(xlim = c(-100, 200), ylim = c(-100, 200)) +
  xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
  ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
  # facet_grid(rows = vars(depth)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())


# # 2001
# sra.01.inc.blk2 %>%
#   filter(d14c_mean.bulk > -100 & d14c_mean.inc > -100) %>%
#   ggplot(., aes(d14c_mean.bulk, d14c_mean.inc, color = pm, shape = eco, group = pm)) +
#   # geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
#   geom_vline(xintercept = 0) +
#   # geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
#   geom_hline(yintercept = 0) +
#   geom_abline(slope = 1, intercept = 0) +
#   geom_point(size = 3) +
#   geom_smooth(method = "lm", formula = y ~ x, se = FALSE) +
#   geom_errorbarh(
#     aes(xmin = d14c_l.bulk, 
#         xmax = d14c_u.bulk,
#         color = pm), 
#     height = 1.5) +
#   geom_errorbar(
#     aes(ymin = d14c_l.inc, 
#         ymax = d14c_u.inc,
#         color = pm), 
#     width = 1.5) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = "blue",
#                                 "basalt" = "red",
#                                 "granite" = "darkgray")) +
#   scale_shape_manual(name = "Ecosystem",
#                      values = c("warm" = 15,
#                                 "cool" = 16,
#                                 "cold" = 17)) +
#   coord_fixed(xlim = c(-100, 200), ylim = c(-100, 200)) +
#   xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
#   ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
#   facet_grid(rows = vars(depth)) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
# 
# # 2019
# fig.n <- fig.n + 1
# sra.19.inc.blk2 %>%
#   mutate(depth = factor(lyr_bot)) %>%
#   ggplot(., aes(d14c_mean.bulk, d14c_mean.inc, color = pm, shape = eco, group = pm)) +
#   # geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
#   geom_vline(xintercept = 0) +
#   # geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
#   geom_hline(yintercept = 0) +
#   geom_abline(slope = 1, intercept = 0) +
#   geom_point(size = 3) +
#   geom_smooth(method = "lm", formula = y ~ x, se = FALSE) +
#   geom_errorbarh(
#     aes(xmin = d14c_l.bulk, 
#         xmax = d14c_u.bulk,
#         color = pm), 
#     height = 1.5) +
#   geom_errorbar(
#     aes(ymin = d14c_l.inc, 
#         ymax = d14c_u.inc,
#         color = pm), 
#     width = 1.5) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = "blue",
#                                 "basalt" = "red",
#                                 "granite" = "darkgray")) +
#   scale_shape_manual(name = "Ecosystem",
#                      values = c("warm" = 0,
#                                 "cool" = 1,
#                                 "cold" = 2)) +
#   coord_fixed(xlim = c(-100, 200), ylim = c(-100, 200)) +
#   xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
#   ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
#   facet_grid(rows = vars(depth)) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())

Fig. 16. Regression of 2019 bulk soil incubations and corresponding bulk soil \(\Delta\)14C

Caption: Regressions of \(\Delta\)14C of bulk soil and respired CO2 by depth for each site in 2019. Error bars show one standard deviation for bulk soil, points show mean of three replicate profiles for bulk soils and single observations for respired CO2.

Time series: \(\Delta\)14C by depth (as measured)

# combine '01, '09, '19 data
sra.01.19.raw <- rbind(bind_rows(sra.2001.sum.ls),
                       bind_rows(sra.2019.sum.ls))
sra.2009.df <- sra.09.sum[ , which(names(sra.09.sum) %in% names(sra.01.19.raw))]
sra.2009.df <- cbind(sra.2009.df, 
                     fm = NA,
                     d14c_sd = NA,
                     fm_sd = NA,
                     d14c_u = NA,
                     d14c_l = NA)
sra.01.09.19.raw <- rbind(sra.01.19.raw, sra.2009.df)
sra.01.09.19.raw$Year <- factor(c(rep(2001, nrow(bind_rows(sra.2001.sum.ls))),
                                  rep(2019, nrow(bind_rows(sra.2019.sum.ls))),
                                  rep(2009, nrow(sra.2009.df))),
                                levels = c("2001", "2009", "2019"))

# plot
# w/ ribbons
# sra.01.09.19.raw %>%
#   mutate(PMeco_year = paste0(PMeco, Year),
#          eco = factor(ifelse(ECO == "pp", "warm",
#                       ifelse(ECO == "wf", "cool", "cold")),
#                       levels = c("warm", "cool", "cold")),
#          d14c_u = d14c + d14c_sd,
#          d14c_l = d14c - d14c_sd,
#          pm = ifelse(PM == "AN", "andesite",
#                      ifelse(PM == "BS", "basalt", "granite"))) %>%
#   ggplot(., aes(d14c, lyr_bot, group = PMeco_year)) +
#   geom_vline(xintercept = 0) +
#   geom_hline(yintercept = 0) +
#   geom_ribbon(aes(xmin = d14c_l, xmax = d14c_u, fill = pm, alpha = Year, group = PMeco_year),
#               color = NA, show.legend = FALSE) +
#   geom_point(aes(fill = pm, color = pm, shape = eco, alpha = Year), size = 2) +
#   geom_point(aes(shape = eco), color = "black", size = 3) +
#   geom_path(aes(linetype = Year, color = pm), size = 0.7) +
#   scale_y_reverse() +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = "blue", 
#                                 "basalt" = "red", 
#                                 "granite" = "darkgray")) +
#   scale_fill_manual(name = "Parent material",
#                     values = c("andesite" = "blue", 
#                                "basalt" = "red", 
#                                "granite" = "darkgray")) +
#   scale_shape_manual(name = "Ecosystem",
#                      values = c("warm" = 22, 
#                                 "cool" = 21, 
#                                 "cold" = 24)) +
#   scale_alpha_manual(values = c("2001" = .6,
#                                 "2009" = 0.4,
#                                 "2019" = 0.2)) +
#   facet_grid(rows = vars(eco), cols = vars(pm)) +
#   xlab(expression(Delta*''^14*'C (‰)')) +
#   ylab("Depth (cm)") +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())

# litter
sra.2019.inc.L.df <- data.frame(
  sra.2019.inc_L.df %>%
    group_by(Year, PM, eco, lyr_bot, PMeco) %>%
    summarize(across(.cols = d14c, 
                     .fns = list(mean = mean, min = min, max = max))) %>%
    rename(year = Year, d14c = d14c_mean) %>%
    mutate(eco = factor(ifelse(eco == "pp", "warm",
                      ifelse(eco == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
           pm = ifelse(PM == "AN", "andesite",
                       ifelse(PM == "BS", "basalt", "granite"))))
# for plotting below
sra.2019.inc.L.df2 <- sra.2019.inc.L.df %>%
  rename(d14c_l = d14c_min,
         d14c_u = d14c_max) %>%
  mutate(PMeco_year = paste0(PMeco, year))

# with error bars, all depths
sra.01.09.19.raw %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  ggplot(., aes(d14c, lyr_bot, group = PMeco_year)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_point(aes(fill = pm, color = pm, shape = eco, alpha = Year), size = 3.5) +
  geom_point(data = sra.2019.inc.L.df2, 
             aes(d14c, lyr_bot, color = pm, shape = eco), shape = 8, size = 3.5, show.legend = FALSE) +
  geom_path(aes(linetype = Year, color = pm), size = 0.7) +
  geom_errorbarh(
    aes(xmin = d14c_l,
        xmax = d14c_u,
        color = pm,
        alpha = Year),
    height = 1.5) +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue",
                                "basalt" = "red",
                                "granite" = "darkgray")) +
  scale_fill_manual(name = "Parent material",
                    values = c("andesite" = "blue",
                               "basalt" = "red",
                               "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15,
                                "cool" = 16,
                                "cold" = 17)) +
  scale_alpha_manual(values = c("2001" = 1,
                                "2009" = 0.6,
                                "2019" = 0.3)) +
  scale_linetype_manual(values = c("2001" = 1,
                                   "2009" = 2,
                                   "2019" = 3)) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())


# just topsoil, w/ error bars
fig.n <- fig.n + 1
sra.01.09.19.raw <- sra.01.09.19.raw[order(sra.01.09.19.raw$lyr_top), ]
sra.01.09.19.raw %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  ggplot(., aes(d14c, lyr_bot, group = PMeco_year)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_point(aes(fill = pm, color = pm, shape = eco, alpha = Year), size = 3) +
  geom_path(aes(linetype = Year, color = pm), size = 0.7) +
  geom_errorbarh(
    aes(xmin = d14c_l,
        xmax = d14c_u,
        color = pm,
        alpha = Year),
    height = 1.5) +
  scale_y_reverse(limits = c(41, 0)) +
  scale_x_continuous(limits = c(-160, 190)) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_fill_manual(name = "Parent material",
                    values = c("andesite" = "blue", 
                               "basalt" = "red", 
                               "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_alpha_manual(values = c("2001" = 1,
                                "2009" = 0.6,
                                "2019" = 0.3)) +
  scale_linetype_manual(values = c("2001" = "solid",
                                   "2009" = "dashed",
                                   "2019" = "dotted")) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. Time series of \(\Delta\)14C by depth, as measured

Caption: Points show mean of three profile replicates for 2001, 2009, and 2019 samples. Error bars show ± 1 standard deviation of the mean (only a single profile was analyzed in 2009). Stars show litter incubation \(\Delta\)14C-CO2 for 2019 samples as a point of reference.

Spline fitting

Soils collected in both the 2001 and 2009 sampling campaigns were sampled by horizon, but the depth intervals differed between the two sampling years. In 2009, full profiles were excavated for each site, as opposed to the shorter profiles collected in 2001 from the GR and AN sites. Radiocarbon was measured on all three replicate profiles at each site for the 2001 samples, but only for one of the replicate profiles at each site in 2009, e.g. ANpp rep2, etc.

In order to compare the radiocarbon profiles between 2001, 2009, and 2019 we first interpolated both radiocarbon and carbon stock data at 1 cm intervals for each site in the datasets from each year. The carbon-stock-weighted radiocarbon values for any given target depth interval can then be calculated as a simple sum of the product of the carbon weight of each 1 cm increment (relative to the total carbon stock of the target depth interval) and its radiocarbon value. A monotonic cubic spline fit with Hyman filtering was used for the carbon stock interpolation (Wendt and Hauser 2013), and a mass-preserving spline was used to fit the radiocarbon data (Bishop, T.F.A., McBratney, A.B., Laslett, G.M., (1999) Modelling soil attribute depth functions with equal-area quadratic smoothing splines. Geoderma, 91(1-2): 27-45).

# 2001
sra.2001.soc.df <- bind_rows(lapply(seq_along(sra.2001.oc.sp.avg), function(i) {
  NM <- names(sra.2001.oc.sp.avg)[i]
  PM <- substr(NM, 1, 2)
  ECO <- substr(NM, 3, 4)
  df <- data.frame(PM = PM, ECO = ECO, lyr_soc_30 = sum(sra.2001.oc.sp.avg[[i]][1:30, "lyr_soc"]))
  df$ECO <- factor(df$ECO, levels = c("pp", "wf", "rf"))
  return(df)
}))

# 2009
sra.2009.soc.df <- bind_rows(lapply(seq_along(sra.2009.oc.sp), function(i) {
  NM <- names(sra.2009.oc.sp)[i]
  PM <- substr(NM, 1, 2)
  ECO <- substr(NM, 3, 4)
  df <- data.frame(PM = PM, ECO = ECO, lyr_soc_30 = sum(sra.2009.oc.sp[[i]][1:30, "lyr_soc"]))
  df$ECO <- factor(df$ECO, levels = c("pp", "wf", "rf"))
  return(df)
}))

# 2019
sra.2019.soc.df <- bind_rows(lapply(seq_along(sra.2019.oc.sp.avg), function(i) {
  NM <- names(sra.2019.oc.sp.avg)[i]
  PM <- substr(NM, 1, 2)
  ECO <- substr(NM, 3, 4)
  df <- data.frame(PM = PM, ECO = ECO, lyr_soc_30 = sum(sra.2019.oc.sp.avg[[i]][1:30, "lyr_soc"]))
  df$ECO <- factor(df$ECO, levels = c("pp", "wf", "rf"))
  return(df)
}))

ggplot(sra.2019.soc.df, aes(PM, lyr_soc_30, fill = PM)) +
  geom_col(position = "dodge") +
  scale_fill_manual(values = c("AN" = andesite, 
                               "BS" = basalt,
                               "GR" = granite)) +
  facet_grid(cols = vars(ECO)) +
  theme_bw() +
  theme(panel.grid = element_blank())


ggplot(sra.2009.soc.df, aes(PM, lyr_soc_30, fill = PM)) +
  geom_col(position = "dodge") +
  scale_fill_manual(values = c("AN" = andesite, 
                               "BS" = basalt,
                               "GR" = granite)) +
  facet_grid(cols = vars(ECO)) +
  theme_bw() +
  theme(panel.grid = element_blank())


ggplot(sra.2001.soc.df, aes(PM, lyr_soc_30, fill = PM)) +
  geom_col(position = "dodge") +
  scale_fill_manual(values = c("AN" = andesite, 
                               "BS" = basalt,
                               "GR" = granite)) +
  facet_grid(cols = vars(ECO)) +
  theme_bw() +
  theme(panel.grid = element_blank())


# all together
sra.01.09.19.soc.df <- cbind(rbind(sra.2001.soc.df,
                                   sra.2009.soc.df,
                                   sra.2019.soc.df), 
                             year = rep(c(2001, 2009, 2019), each = 9))
sra.01.09.19.soc.df %>%
  mutate(PMyear = paste0(PM, year)) %>%
  ggplot(., aes(PMyear, lyr_soc_30, fill = PM, alpha = year)) +
  geom_col(position = "dodge") +
  scale_fill_manual(values = c("AN" = andesite, 
                               "BS" = basalt,
                               "GR" = granite)) +
  facet_grid(cols = vars(ECO)) +
  theme_bw() +
  theme(panel.grid = element_blank())

#### 0-30cm
### bulk
## 2019
# SOC weights
cwt.19_30 <- lapply(seq_along(sra.2019.oc.sp), function(i) {
  lapply(sra.2019.oc.sp[[i]], function(df) {
    d <- 30
    c <- df[1:d, "lyr_soc"]
    return(unlist(lapply(c, function(x) x/sum(c))))
  })
})
names(cwt.19_30) <- names(sra.2019.oc.sp)
# FM wts
fm.wt.19_30 <- lapply(seq_along(cwt.19_30), function(i) {
  lapply(seq_along(cwt.19_30[[i]]), function(j) {
    df <- data.frame(cwt = cwt.19_30[[i]][[j]])
    df$fm <- sra.2019.fm.sp[[i]][[j]][["var.1cm"]][1:length(cwt.19_30[[i]][[j]])]
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.19_30) <- names(cwt.19_30)
# summarize over 0-30cm
sra.19.rep.30.ls <- lapply(seq_along(1:9), function(i) {
  lapply(seq_along(fm.wt.19_30[[i]]), function(x) {
    d <- 30
    f <- sum(fm.wt.19_30[[i]][[x]][1:d, "fm_wt"])
    return(unlist(f)) 
  })
})
sra.19.avg.30.ls <- lapply(seq_along(sra.2019.sum.ls), function(i) {
  fm <- lapply(seq_along(sra.19.rep.30.ls[[i]][[1]]), function(z) {
    data.frame(fm_19_mean = mean(sapply(sra.19.rep.30.ls[[i]], "[", z), na.rm = TRUE),
               fm_19_sd = sd(sapply(sra.19.rep.30.ls[[i]], "[", z), na.rm = TRUE))
    })
  return(bind_rows(fm))
})
names(sra.19.avg.30.ls) <- names(fm.wt.19_30)

## 2001
# SOC weights
cwt.01_30 <- lapply(sra.2001.oc.sp, function(ls) {
  lapply(ls, function(df) {
    d <- 30
    c <- df[1:d, "lyr_soc"]
    return(unlist(lapply(c, function(x) x/sum(c))))
  })
})
names(cwt.01_30) <- names(sra.2001.oc.sp)
# FM wts
fm.wt.01_30 <- lapply(seq_along(cwt.01_30), function(i) {
  lapply(seq_along(cwt.01_30[[i]]), function(j) {
    df <- data.frame(cwt = cwt.01_30[[i]][[j]])
    df$fm <- sra.2001.fm.sp[[i]][[j]][["var.1cm"]][1:length(cwt.01_30[[i]][[j]])]
    # linear extrapolation for filling 20-30cm fm data
    fm_1_30 <- df$fm[1:30] # 0-30cm fm
    if(length(which(is.na(fm_1_30))) > 0) {
     ix <- which(is.na(fm_1_30))
     ix.min <- min(ix) # first is.na(fm)
     m <- fm_1_30[ix.min-1]-fm_1_30[ix.min-2] # slope at last two measurement points
     for(i in ix.min:30) {
      fm_1_30[i] <- fm_1_30[i - 1] + m 
     }
     df$fm[1:30] <- fm_1_30 
    }
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.01_30) <- names(cwt.01_30)
# summarize over 0-30cm
sra.01.rep.30.ls <- lapply(seq_along(1:9), function(i) {
  lapply(seq_along(fm.wt.01_30[[i]]), function(x) {
    d <- 30
    f <- sum(fm.wt.01_30[[i]][[x]][1:d, "fm_wt"])
    return(unlist(f)) 
  })
})
sra.01.avg.30.ls <- lapply(seq_along(1:9), function(i) {
  fm <- lapply(seq_along(sra.01.rep.30.ls[[i]][[1]]), function(z) {
    data.frame(fm_01_mean = mean(sapply(sra.01.rep.30.ls[[i]], "[", z), na.rm = TRUE),
               fm_01_sd = sd(sapply(sra.01.rep.30.ls[[i]], "[", z), na.rm = TRUE))
    })
  return(bind_rows(fm))
})
names(sra.01.avg.30.ls) <- names(fm.wt.01_30)

### inc
## 2019
# SOC weights (site average)
cwt.19_30.avg <- lapply(cwt.19_30, function(ls) {
  apply(bind_rows(ls), 1, mean)
})
# FM weights
fm.wt.19.30.inc <- lapply(seq_along(cwt.19_30.avg), function(j) {
  lapply(sra.2019.inc.fm.sp[[j]], function(fm) {
    df <- data.frame(cwt = cwt.19_30.avg[[j]])
    df$fm <- fm
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.19.30.inc) <- names(cwt.19_30.avg)
# summarize over 0-30cm
sra.19.30.inc.ls <- lapply(fm.wt.19.30.inc, function(ls) {
  ls <- lapply(ls, function(df) sum(df$fm_wt))
  names(ls) <- c("fm_19_mean", "fm_19_min", "fm_19_max")
  return(data.frame(bind_rows(ls)))
})
names(sra.19.30.inc.ls) <- names(cwt.19_30.avg)

## 2001
# SOC weights (site average)
cwt.01_30.avg <- lapply(cwt.01_30, function(ls) {
  apply(bind_rows(ls), 1, mean)
})
# FM weights
fm.wt.01.30.inc <- lapply(seq_along(cwt.01_30.avg), function(j) {
  lapply(sra.2001.inc.fm.sp[[j]], function(fm) {
    df <- data.frame(cwt = cwt.01_30.avg[[j]])
    df$fm <- fm[1:length(cwt.01_30.avg[[j]])]
    # linear extrapolation for filling 20-30cm fm data
    fm_1_30 <- df$fm[1:30] # 0-30cm fm
    if(length(which(is.na(fm_1_30))) > 0) {
     ix <- which(is.na(fm_1_30))
     ix.min <- min(ix) # first is.na(fm)
     m <- fm_1_30[ix.min-1]-fm_1_30[ix.min-2] # slope at last two measurement points
     for(i in ix.min:30) {
      fm_1_30[i] <- fm_1_30[i - 1] + m 
     }
     df$fm[1:30] <- fm_1_30 
    }
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.01.30.inc) <- names(cwt.01_30.avg)
# summarize over 0-30cm
sra.01.30.inc.ls <- lapply(fm.wt.01.30.inc, function(ls) {
  ls <- lapply(ls, function(df) sum(df$fm_wt))
  names(ls) <- c("fm_01_mean", "fm_01_min", "fm_01_max")
  return(data.frame(bind_rows(ls)))
})
names(sra.01.30.inc.ls) <- names(cwt.01_30.avg)

## df for linear modeling
# bulk
sra.blk.rep.30.ls <- lapply(seq_along(sra.01.rep.30.ls), function(i) {
  blk.01 <- data.frame(fm_blk = do.call(rbind, sra.01.rep.30.ls[[i]]),
                       year = 2001)
  blk.19 <- data.frame(fm_blk = do.call(rbind, sra.19.rep.30.ls[[i]]),
                       year = 2019)
  rbind(blk.01, blk.19) %>%
    mutate(d14c_blk = calc_14c(fm_blk, year))
})
names(sra.blk.rep.30.ls) <- names(sra.01.30.inc.ls)
sra.blk.rep.30.df <- bind_rows(sra.blk.rep.30.ls, .id = "PMeco")
# inc
inc.30.bind.fx <- function(ls, year_xx) {
  rbind(
    bind_rows(lapply(ls, "[", 2), .id = "PMeco") %>%
      rename(fm_inc = paste0("fm_", year_xx, "_min")),
    bind_rows(lapply(ls, "[", 3), .id = "PMeco") %>%
      rename(fm_inc = paste0("fm_", year_xx, "_max"))) %>%
    mutate(year = as.numeric(paste0("20", year_xx))) %>%
    mutate(d14c_inc = calc_14c(fm_inc, year))
}
sra.inc.rep.30.df <- rbind(inc.30.bind.fx(sra.01.30.inc.ls, "01"), 
                           inc.30.bind.fx(sra.19.30.inc.ls, "19"))
# combine
sra.blk.inc.rep.30.df <- merge(sra.blk.rep.30.df, sra.inc.rep.30.df, by = c("year", "PMeco"))
save(sra.blk.inc.rep.30.df, file = "sra.blk.inc.rep.30.df.RData")

## Combine mean data into a single data frame
# functions for converting fm to d14c and calculating sd
blk.14c.sd.fx <- function(df, year_xx) {
  date <- as.numeric(paste0(20, year_xx))
  df$fm_u <- df[[paste0("fm_", year_xx, "_mean")]] + df[[paste0("fm_", year_xx, "_sd")]]
  df$d14c_u <- calc_14c(df$fm_u, date)
  df[[paste0("d14c_", "mean")]] <- calc_14c(df[[paste0("fm_", year_xx, "_mean")]], date)
  df[[paste0("d14c_", "sd")]] <- df[[paste0("d14c_", "mean")]] - df$d14c_u
  df$year <- as.numeric(paste0(20, year_xx))
  return(df %>% select(c(starts_with("d14c"), year)) %>% select(-d14c_u))
}
inc.14c.sd.fx <- function(df, year_xx) {
  names(df) <- gsub(paste0("fm_", year_xx), "d14c", names(df))
  df_14c <- calc_14c(df, as.numeric(paste0(20, year_xx)))
  df_14c[[paste0("d14c_", "sd")]] <- sd(df_14c[ , 2:3])
  df_14c$year <- as.numeric(paste0(20, year_xx))
  return(df_14c[ , c(1, 4:5)])
}
# run functions and combine lists 
# 0-30cm data from '01 and '19
sra.30.blk.inc.ls <- lapply(
  list(lapply(sra.01.30.inc.ls, inc.14c.sd.fx, year_xx = "01"), 
       lapply(sra.19.30.inc.ls, inc.14c.sd.fx, year_xx = "19"),
       lapply(sra.01.avg.30.ls, blk.14c.sd.fx, year_xx = "01"),
       lapply(sra.19.avg.30.ls, blk.14c.sd.fx, year_xx = "19")),
  bind_rows, .id = "PMeco")
# reduce list to data frame, calculate difference of means and sd
sra.30.blk.inc.df <- rbind(merge(sra.30.blk.inc.ls[[1]],
                                 sra.30.blk.inc.ls[[3]],
                                 by = c("PMeco", "year"), suffixes = c("_inc", "_blk")),
                           merge(sra.30.blk.inc.ls[[2]],
                                 sra.30.blk.inc.ls[[4]],
                                 by = c("PMeco", "year"), suffixes = c("_inc", "_blk"))) %>%
  mutate(blk.inc = d14c_mean_blk - d14c_mean_inc,
         blk.inc.sd = sqrt(d14c_sd_blk^2/3 + d14c_sd_inc^2/2))
fig.n <- fig.n + 1
sra.01.09.19 %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  # filter(Year != "2009") %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, linetype = Year, group = PMeco_year)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_point(aes(alpha = Year), size = 3) +
  geom_path(aes(linetype = Year)) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  scale_y_reverse() +
  scale_x_continuous() +    
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_alpha_manual(values = c("2001" = 1,
                                "2009" = 0.6,
                                "2019" = 0.3)) +
  scale_linetype_manual(values = c("2001" = "solid",
                                 "2009" = "dashed",
                                 "2019" = "dotted")) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. Time series of bulk soil \(\Delta\)14C by 2001 depths (2001, 2009, 2019 samples)

Caption: Points for 2001 samples show the mean \(\Delta\)14C values at the measured depths. Points for 2009 and 2019 samples are spline-fitted estimates of \(\Delta\)14C predicted for the same depth intervals as measured in 2001. Error bars show ± 1 standard deviation of the mean of three replicate profiles for 2001 and 2019 samples (only a single profile was analyzed in 2009).

fig.n <- fig.n + 1
sra.19.01.09 %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, linetype = Year, group = PMeco_year)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_point(aes(alpha = Year), size = 3) +
  geom_path(aes(linetype = Year)) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  # scale_y_reverse(limits = c(30, 0)) +
  scale_y_reverse() +
  scale_x_continuous() +    
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_alpha_manual(values = c("2001" = 1,
                                "2009" = 0.6,
                                "2019" = 0.3)) +
  scale_linetype_manual(values = c("2001" = "solid",
                                   "2009" = "dashed",
                                   "2019" = "dotted")) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Fig. 16. Time series of bulk soil \(\Delta\)14C by depth (splined to 2019 depths)

Caption: Points for 2019 samples show the mean \(\Delta\)14C values at the measured depths. Points for 2001 and 2009 samples are spline-fitted estimates of \(\Delta\)14C predicted for the same depth intervals as measured in 2019. Error bars show ± 1 standard deviation of the mean of three replicate profiles for 2001 and 2019 samples (only a single profile was analyzed in 2009). NB: Only two depth intervals were measured at the cool and cold andesite sites (max depth of 27 and 28 cm, respectively), so linear extrapolation (using the slope of the last 1cm spline-fitted depth increment) was used to extend the profiles to 30 cm.

# plot individual depths
fig.n <- fig.n + 1

# Atm
atm.14c <- data.frame(year = Datm[Datm$Date > 2000, "Date"],
                      d14c = Datm[Datm$Date > 2000, "NHc14"])
save(atm.14c, file = "atm.14c.RData")

# bulk 14C over time for 0-10, 10-20, 20-30 w/ atm
sra.19.01.09 %>%
  filter(lyr_bot < 31) %>%
  mutate(PMeco_depth = paste0(PMeco, lyr_bot),
         depth = factor(lyr_bot),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite")),
         year = as.numeric(as.character(Year))) %>%
  ggplot(., aes(year, d14c)) +
  geom_path(data = atm.14c) +
  geom_point(aes(color = pm, shape = eco), size = 3) +
  geom_path(aes(color = pm, group = PMeco_depth, linetype = depth), alpha = 0.3) +
  geom_errorbar(
    aes(ymin = d14c_l, 
        ymax = d14c_u,
        color = pm), 
    width = .5) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_linetype_manual(name = "Depth (cm)",
                        labels = c("10" = "0-10",
                                   "20" = "10-20",
                                   "30" = "20-30"),
                        values = c("10" = 1,
                                   "20" = 2,
                                   "30" = 3)) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  ylab(expression(Delta*''^14*'C (‰)')) +
  xlab("Year") +
  theme_bw() +
  theme(panel.grid = element_blank())


### incubation
## 2019
sra.2019.inc.df <- bind_rows(lapply(sra.2019.inc.ls, function(df) {
  data.frame(df %>%
               group_by(Year, PM, ECO, lyr_bot, PMeco) %>%
               summarize(
                 across(.cols = d14c, 
                        .fns = list(mean = mean, min = min, max = max))) %>%
               rename(year = Year, d14c = d14c_mean))
}))
save(sra.2019.inc.df, file = "sra.2019.inc.df.RData")
## 2001
sra.19.01.inc.df <- bind_rows(lapply(seq_along(sra.19.01.inc.ls), function(i) {
  PMeco <- names(sra.19.01.inc.ls)[i]
  d14c.ls <- lapply(sra.19.01.inc.ls[[i]], calc_14c, obs_date_y = 2001)
  df <- data.frame(d14c = d14c.ls[[1]],
                   d14c_min = d14c.ls[[2]],
                   d14c_max = d14c.ls[[3]],
                   lyr_bot = c(10, 20, 30),
                   PMeco = PMeco,
                   PM = substr(PMeco, 1, 2),
                   ECO = substr(PMeco, 3, 4),
                   year = 2001)
  return(df)
}))
# join
sra.19.01.inc <- rbind(sra.19.01.inc.df, sra.2019.inc.df)

# plot
sra.19.01.inc %>%
  mutate(PMeco_depth = paste0(PMeco, lyr_bot),
         depth = factor(lyr_bot),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  ggplot(., aes(year, d14c)) +
  geom_path(data = atm.14c) +
  geom_point(aes(color = eco, shape = eco), size = 3) +
  geom_point(data = sra.2019.inc.L.df, aes(color = eco), shape = 8, size = 3, show.legend = FALSE) +
  geom_path(aes(color = eco, group = PMeco), alpha = 0.3) +
  geom_errorbar(
    aes(ymin = d14c_min, 
        ymax = d14c_max,
        color = eco), 
    width = .5) +
  geom_errorbar(
    data = sra.2019.inc.L.df,
    aes(ymin = d14c_min, 
        ymax = d14c_max,
        color = eco), 
    width = .5) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_y_continuous(limits = c(-40, 170)) +
  facet_grid(rows = vars(pm), cols = vars(depth)) +
  ylab(expression(Delta*''^14*'C (‰)')) +
  xlab("Year") +
  theme_bw() +
  theme(panel.grid = element_blank())


# plot inc and bulk together, by depth
sra.ts.all <- sra.19.01.09 %>%
  filter(lyr_bot < 31) %>%
  select(Year, PM, ECO, PMeco, lyr_bot, d14c, d14c_sd) %>%
  mutate(Type = "bulk",
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         year = as.numeric(as.character(Year))) %>%
  select(-d14c_sd, -Year) %>%
  bind_rows(.,
            sra.19.01.inc %>%
              select(year, PM, ECO, PMeco, lyr_bot, d14c, d14c_min, d14c_max) %>%
              rename(d14c_l = d14c_min,
                     d14c_u = d14c_max) %>%
              mutate(Type = "inc")
  ) %>%
  mutate(depth = factor(lyr_bot),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite")),
         ecoType = paste0(eco, " (", Type, ")"))

# Plot by depth
plot.ts.fx <- function(df) {
  df %>%
    filter(d14c > -200) %>%
    filter(year != 2009) %>%
    ggplot(., aes(year, d14c)) +
    geom_path(data = atm.14c) +
    geom_point(aes(color = pm, shape = ecoType), size = 3) +
    geom_path(aes(color = pm, linetype = Type), alpha = 0.3) +
    geom_errorbar(
      aes(ymin = d14c_l, 
          ymax = d14c_u,
          color = pm), 
      width = .5) +
    scale_color_manual(name = "Parent material",
                       values = c("andesite" = "blue", 
                                  "basalt" = "red", 
                                  "granite" = "darkgray")) +
    scale_shape_manual(name = "Ecosystem (type)",
                       values = c("warm (inc)" = 0,
                                  "cool (inc)" = 1,
                                  "cold (inc)" = 2,
                                  "warm (bulk)" = 15,
                                  "cool (bulk)" = 16,
                                  "cold (bulk)" = 17)) +
    facet_grid(rows = vars(eco), cols = vars(pm)) +
    ylab(expression(Delta*''^14*'C (‰)')) +
    xlab("Year") +
    theme_bw() +
    theme(panel.grid = element_blank())
}

# plots
lapply(split(sra.ts.all, sra.ts.all$depth), plot.ts.fx)
$`10`

$`20`

$`30`

# # to save
# for(i in 1:3) ggsave(paste0(i, ".pdf"), lapply(split(sra.ts.all, sra.ts.all$depth), plot.ts.fx)[[i]])

Fig. 16. Change in \(\Delta\)14C of bulk soil (panel a) and respired CO2 (panel b) over time relative to the atmosphere

Caption: Points for 2019 samples show the mean \(\Delta\)14C values at the measured depths. Points for 2001 and 2009 (bulk only) samples are spline-fitted estimates of \(\Delta\)14C predicted for the same depth intervals as measured in 2019. Error bars for bulk samples in panel (a) show ± 1 standard deviation of the mean of three replicate profiles for 2001 and 2019 samples (only a single profile was analyzed in 2009); error bars for incubation samples in panel (b) show the values of the two reps, while the point represents the mean. NB: Only two depth intervals were measured at the cool and cold andesite sites (max depth of 27 and 28 cm, respectively), so linear extrapolation (using the slope of the last 1cm spline-fitted depth increment) was used to extend the profiles to 30 cm.

# load data
ras18.frc <- read_excel("/Users/jeff/sra-ts/data/external/sra_ras_sum/sierra_data_summary_2020.xlsx",
                        sheet = "2009_fraction_data")

# select only min cols and pivot longer
ras18_2 <- ras18.sum %>%
  select(`Fed (g/kg)`, `Feo (g/kg)`, `Alo (g/kg)`, `Alp (g/kg)`, `top mineral`, `bottom mineral`, pro_name) %>% 
  rename(lyr_top = `top mineral`,
         lyr_bot = `bottom mineral`) %>%
  pivot_longer(cols = c(`Fed (g/kg)`, 
                        `Feo (g/kg)`, 
                        `Alo (g/kg)`, 
                        `Alp (g/kg)`), 
               names_to = "mins", values_to = "conc") %>%
  data.frame()

# Calculate min stocks
ras18_3 <- ras18.sum %>%
  select(`Fed (g/kg)`, `Feo (g/kg)`, `Alo (g/kg)`, `Alp (g/kg)`, `top mineral`, `bottom mineral`, pro_name, BD_g_cm_3, Soil_finefraction, Thickness_cm) %>% 
  rename(lyr_top = `top mineral`,
         lyr_bot = `bottom mineral`) %>%
  pivot_longer(cols = c(`Fed (g/kg)`, 
                        `Feo (g/kg)`, 
                        `Alo (g/kg)`, 
                        `Alp (g/kg)`), 
               names_to = "mins", values_to = "conc") %>%
  mutate(mass = Thickness_cm * BD_g_cm_3 * Soil_finefraction * 10,
         min_stock = conc * mass * 10^-2) %>%
  data.frame()
ras18_3.ls <- lapply(split(ras18_3, ras18_3$mins), function(df) {
  lapply(split(df, df$pro_name), function(x) {
    x <- x[order(x$lyr_bot), ]
    # calc cmtv min stock
    x$min_stock_cmtv <- NA
    for(i in seq_along(x$lyr_bot)) {
      if(i == 1) {
        x$min_stock_cmtv[i] <- x$min_stock[i]
      } else {
        x$min_stock_cmtv[i] <- x$min_stock[i] + x$min_stock_cmtv[i-1] 
      }
    }
    x$mass_cmtv <- NA
    for(i in seq_along(x$lyr_bot)) {
      if(i == 1) {
        x$mass_cmtv[i] <- x$mass[i]
      } else {
        x$mass_cmtv[i] <- x$mass[i] + x$mass_cmtv[i-1] 
      }
    }
    return(x)
  })
})
ras18_3.sp.df <- bind_rows(lapply(ras18_3.ls, function(ls) {
  bind_rows(lapply(ls, function(x) {
    depths(x) <- pro_name ~ lyr_top + lyr_bot
    x.mps <- mpspline(x, var.name = "min_stock_cmtv")
    return(x.mps$var.1cm[30])
  }), .id = "pro_name") %>%
    pivot_longer(cols = everything(), names_to = "pro_name", values_to = "min_stock_cmtv")
}), .id = "min") %>%
  mutate(min = ifelse(min == "Alo (g/kg)", "Al_ox",
                      ifelse(min == "Alp (g/kg)", "Al_py",
                             ifelse(min == "Fed (g/kg)", "Fe_dc", "Fe_ox"))),
         PMeco = substr(pro_name, 1, 4)) %>%
  select(-pro_name)
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
# mass-weighted concentration
ras18_4.sp.ls <- lapply(ras18_3.ls, function(ls) {
  conc <- unlist(lapply(ls, function(x) {
    depths(x) <- pro_name ~ lyr_top + lyr_bot
    x.mps <- mpspline(x, var.name = "conc")
    df <- data.frame(conc = x.mps$var.1cm[1:30],
                     lyr_bot = seq(1, 30),
                     pro_name = substr(x.mps$idcol, 1, 4))
    return(split(df, df$pro_name))
    }), recursive = FALSE)
  mass <- unlist(lapply(seq_along(ls), function(i) {
    x <- ls[[i]][ , c("lyr_bot", "mass_cmtv")]
    t0 <- data.frame(matrix(nrow = 1, ncol = ncol(x)))
    names(t0) <- names(x)
    t0 <- 0
    x <- rbind(t0, x)
    sp <- spline(x, method = "hyman") # fit monotonic cubic spline
    sp.ss <- smooth.spline(sp) # convert to class "spline" with smooth.spline fxn
    std <- seq(0, 30) # depth in cm
    sp <- predict(sp.ss, std) 
    df <- data.frame(sp)
    colnames(df) <- c("lyr_bot", "mass_cmtv")
    df$pro_name <- substr(names(ls)[i], 1, 4)
    df <- df[-1, ]
    return(split(df, df$pro_name))
  }), recursive = FALSE)
  return(mapply(merge,
                mass,
                conc,
                SIMPLIFY = FALSE))
})
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
# calculate mass-weighted conc for 0-30cm
ras18_4.sp.df <- bind_rows(lapply(ras18_4.sp.ls, function(ls) {
  bind_rows(lapply(ls, function(df) {
    df <- df[order(df$lyr_bot), ]
    df$mass <- NA
    for (i in seq_along(df$mass)) {
      if (i == 1) {
        df$mass[i] <- df$mass_cmtv[i]
      } else {
        df$mass[i] <- df$mass_cmtv[i] - df$mass_cmtv[i-1]
      }
    }
    df$mass_wt <- df$mass/sum(df$mass)
    df$conc_30_wtd <- df$mass_wt * df$conc
    return(sum(df$conc_30_wtd))
  }), .id = "pro_name")
}), .id = "min") %>%
  pivot_longer(!min, names_to = "PMeco", values_to = "conc") %>%
  mutate(min = ifelse(min == "Alo (g/kg)", "Al_ox", 
                      ifelse(min == "Alp (g/kg)", "Al_py",
                             ifelse(min == "Fed (g/kg)", "Fe_dc", "Fe_ox"))))

# merge w/ 14C data
sra.all.30.min.conc.wtd <- merge(sra.30.blk.inc.df, ras18_4.sp.df, by = "PMeco") %>%
  mutate(pm = ifelse(substr(PMeco, 1, 2) == "AN", "andesite", 
                     ifelse(substr(PMeco, 1, 2) == "BS", "basalt", "granite")),
         eco = ifelse(substr(PMeco, 3, 4) == "pp", "warm", 
                      ifelse(substr(PMeco, 3, 4) == "wf", "cool", "cold")))
save(sra.all.30.min.conc.wtd, file = "sra.all.30.min.conc.wtd.RData")

# spline fits
# (should be mass-weighted...)
# also calculate for 0-30cm
ras18.split <- split(ras18_2, ras18_2$mins)
ras18.sp <- lapply(ras18.split, function(df) {
  ls <- lapply(split(df, df$pro_name), function(x) {
    depths(x) <- pro_name ~ lyr_top + lyr_bot
    x.mps <- mpspline(x, var.name = "conc", d = t(seq(0, 100, 10)))
    return(x.mps$var.std)
  })
  names(ls) <- unique(df$pro_name)
  return(ls)
})
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
Fitting mass preserving splines per profile...

  |                                                                                               
  |                                                                                         |   0%
  |                                                                                               
  |=========================================================================================| 100%
names(ras18.sp) <- c("Al_ox", "Al_py", "Fe_dc", "Fe_ox")
ras18.sp.df <- data.frame(reduce(lapply(seq_along(ras18.sp), function(i) {
    df <- data.frame(t(bind_rows(ras18.sp[[i]])))
    names(df) <- unique(ras18.sum$pro_name)
    df$depth <- rownames(df)
    return(df %>%
             pivot_longer(!depth, names_to = "pro_name", values_to = names(ras18.sp)[i]))
  }),
  left_join,
  by = c("depth", "pro_name")
))
ras18.sp.df <- ras18.sp.df[-which(ras18.sp.df$depth == "soil depth"), ]
ras18.sp.df$lyr_bot <- rep(seq(10, 100, 10), each = 9)
ras18.sp.df <- ras18.sp.df[complete.cases(ras18.sp.df), ]
ras18.sp.df$PM <- substr(ras18.sp.df$pro_name, 1, 2)
ras18.sp.df$ECO <- substr(ras18.sp.df$pro_name, 3, 4)
save(ras18.sp.df, file = "ras18.sp.df.RData")

# reshape sra.ts.all w/ bulk and inc in separate cols
nms.inc.blk2 <- nms.inc.blk
nms.inc.blk2[[4]] <- "year"
sra.ts.all.blk.inc <- merge(sra.ts.all[sra.ts.all$Type == "bulk", ],
                            sra.ts.all[sra.ts.all$Type == "inc", c(nms.inc.blk2, "d14c", "d14c_u", "d14c_l")],
                            by = nms.inc.blk2,
                            suffixes = c("_bulk", "_inc")) %>%
  filter(year != 2009) %>%
  mutate(blk.inc = d14c_bulk - d14c_inc,
         blk.inc.sd = sqrt((d14c_u_bulk - d14c_bulk)^2 + apply(cbind(d14c_u_inc, d14c_l_inc), 1, var)))

# join w/ d14c
sra.all.min <- ras18.sp.df %>%
  mutate(pm = ifelse(PM == "AN", "andesite", ifelse(PM == "BS", "basalt", "granite")),
         eco = ifelse(ECO == "pp", "warm", ifelse(ECO == "wf", "cool", "cold"))) %>%
  # mutate(Al_nonCrys = Al_ox - Al_py,
  #        Fe_Crys = Fe_dc - Fe_ox) %>%
  select(-PM, -ECO, -pro_name) %>%
  left_join(sra.ts.all.blk.inc[ , c("pm", "eco", "lyr_bot", "year", "d14c_bulk", "d14c_u_bulk", "d14c_inc", "d14c_u_inc", "d14c_l_inc", "d14c_l_bulk", "blk.inc", "blk.inc.sd")], 
            ., 
            by = c("pm", "eco", "lyr_bot")) %>%
  pivot_longer(cols = c("Al_py", "Al_ox", "Fe_ox", "Fe_dc", 
                        # "Al_nonCrys", "Fe_Crys"
                        ), names_to = "min", values_to = "conc")

# Create min/14c df w/ 0-30cm 14C data
sra.all.30.min <- merge(sra.30.blk.inc.df, ras18_3.sp.df, by = "PMeco") %>%
  mutate(pm = ifelse(substr(PMeco, 1, 2) == "AN", "andesite", 
                     ifelse(substr(PMeco, 1, 2) == "BS", "basalt", "granite")),
         eco = ifelse(substr(PMeco, 3, 4) == "pp", "warm", 
                      ifelse(substr(PMeco, 3, 4) == "wf", "cool", "cold")))


# save
save(sra.all.min, file = "sra.all.min.RData")
save(sra.all.30.min, file = "sra.all.30.min.RData")
# bulk
sra.all.min %>%
  mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
         ecoYear = paste0(eco, " (", year, ")"),
         width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
  filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  ggplot(., aes(conc, d14c_bulk)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = d14c_l_bulk,
        ymax = d14c_u_bulk,
        color = pm,
        width = width)) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())


# inc
sra.all.min %>%
  mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
         ecoYear = paste0(eco, " (", year, ")"),
         width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
  filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  filter(lyr_bot == 30) %>%
  ggplot(., aes(conc, d14c_inc)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = d14c_l_inc,
        ymax = d14c_u_inc,
        color = pm,
        width = width)) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())


# bulk-inc
sra.all.min %>%
  mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
         ecoYear = paste0(eco, " (", year, ")"),
         width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
  filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  ggplot(., aes(conc, blk.inc)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = blk.inc - blk.inc.sd,
        ymax = blk.inc + blk.inc.sd,
        color = pm,
        width = width)) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())


# Fe_dc alone
#####
# warm
sra.all.min %>%
  filter(eco == "warm" & min == "Fe_dc") %>%
  mutate(ecoYear = paste0(eco, " (", year, ")")) %>%
  ggplot(., aes(conc, blk.inc)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = blk.inc - blk.inc.sd,
        ymax = blk.inc + blk.inc.sd,
        color = pm),
        width = 1.5) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15,
                                "warm (2019)" = 0)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())


# cool + cold
sra.all.min %>%
  filter(eco != "warm" & min == "Fe_dc") %>%
  mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
         ecoYear = paste0(eco, " (", year, ")")) %>%
  filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  ggplot(., aes(conc, blk.inc)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = blk.inc - blk.inc.sd,
        ymax = blk.inc + blk.inc.sd,
        color = pm),
        width = .5) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

#####

## by depth
#####
# # 10 cm
# sra.all.min %>%
#   mutate(ecoYear = paste0(eco, " (", year, ")"),
#          width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
#   filter(lyr_bot == 10) %>%
#   ggplot(., aes(conc, blk.inc)) +
#   geom_point(aes(color = pm, shape = ecoYear), size = 3) +
#   geom_errorbar(
#     aes(ymin = blk.inc - blk.inc.sd,
#         ymax = blk.inc + blk.inc.sd,
#         color = pm,
#         width = width)) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = andesite,
#                                 "basalt" = basalt,
#                                 "granite" = granite)) +
#   scale_shape_manual(name = "Climate (year)",
#                      values = c("warm (2001)" = 15, 
#                                 "cool (2001)" = 16, 
#                                 "cold (2001)" = 17,
#                                 "warm (2019)" = 0, 
#                                 "cool (2019)" = 1, 
#                                 "cold (2019)" = 2)) +
#   facet_wrap(vars(min), scales = "free") +
#   ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
#   xlab(expression('Concentration (g kg'^-1*')')) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
# 
# # 20 cm
# sra.all.min %>%
#   mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
#          ecoYear = paste0(eco, " (", year, ")"),
#          width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
#   filter(pmEcoDepth != "granitecold20") %>%
#   filter(lyr_bot == 20) %>%
#   ggplot(., aes(conc, blk.inc)) +
#   geom_point(aes(color = pm, shape = ecoYear), size = 3) +
#   geom_errorbar(
#     aes(ymin = blk.inc - blk.inc.sd,
#         ymax = blk.inc + blk.inc.sd,
#         color = pm,
#         width = width)) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = andesite,
#                                 "basalt" = basalt,
#                                 "granite" = granite)) +
#   scale_shape_manual(name = "Climate (year)",
#                      values = c("warm (2001)" = 15, 
#                                 "cool (2001)" = 16, 
#                                 "cold (2001)" = 17,
#                                 "warm (2019)" = 0, 
#                                 "cool (2019)" = 1, 
#                                 "cold (2019)" = 2)) +
#   facet_wrap(vars(min), scales = "free") +
#   ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
#   xlab(expression('Concentration (g kg'^-1*')')) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
# 
# # 30 cm
# sra.all.min %>%
#   mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
#          ecoYear = paste0(eco, " (", year, ")"),
#          width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
#   filter(pmEcoDepth != "granitecold30") %>%
#   filter(lyr_bot == 30) %>%
#   ggplot(., aes(conc, blk.inc)) +
#   geom_point(aes(color = pm, shape = ecoYear), size = 3) +
#   geom_errorbar(
#     aes(ymin = blk.inc - blk.inc.sd,
#         ymax = blk.inc + blk.inc.sd,
#         color = pm,
#         width = width)) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = andesite,
#                                 "basalt" = basalt,
#                                 "granite" = granite)) +
#   scale_shape_manual(name = "Climate (year)",
#                      values = c("warm (2001)" = 15, 
#                                 "cool (2001)" = 16, 
#                                 "cold (2001)" = 17,
#                                 "warm (2019)" = 0, 
#                                 "cool (2019)" = 1, 
#                                 "cold (2019)" = 2)) +
#   facet_wrap(vars(min), scales = "free") +
#   ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
#   xlab(expression('Concentration (g kg'^-1*')')) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
# function for Tukey HSD tables
tukey.table.fx <- function(x, year, type, var) {
  depth <- paste0(unique(x$lyr_bot) - 10, "-", unique(x$lyr_bot), " cm")
  if (type == "inc") {
    x <- x[x$d14c > -200, c("d14c", var)]
  } 
  return(
    TukeyHSD(aov(reformulate(var, "d14c"), x))[var] %>%
    data.frame(.) %>%
    mutate(Pairs = rownames(.)) %>%
    mutate(across(where(is.numeric), round, 3)) %>%
    gt() %>%
    tab_header(
      title = depth,
      subtitle = paste(year, type, var)
    ))
}

### 2001
## bulk
sra.2001.bulk.df <- bind_rows(
  lapply(sra.19.01.rep.ls, function(ls) {
    ls <- lapply(ls, function(x) x[complete.cases(x)])
    d14c <- calc_14c(unlist(ls), 2001)
    df <- data.frame(d14c = d14c,
                     lyr_bot = rep(c(10, 20, 30), length(d14c) / 3))
    return(df)
  }),
  .id = "PMeco") %>%
  mutate(PM = substr(PMeco, 1, 2),
         ECO = substr(PMeco, 3, 4))
# PM
# lapply(split(sra.2001.bulk.df, sra.2001.bulk.df$lyr_bot), function(x) {
#   summary(lm(d14c ~ PM, x))
# })
lapply(split(sra.2001.bulk.df, sra.2001.bulk.df$lyr_bot), function(x) {
  tukey.table.fx(x, "2001", "bulk", "PM")
})
$`10`
0-10 cm
2001 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
3.776 -57.040 64.592 0.987 BS-AN
43.034 -17.782 103.850 0.201 GR-AN
39.258 -19.742 98.258 0.239 GR-BS

$`20`
10-20 cm
2001 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
47.757 4.712 90.803 0.028 BS-AN
45.292 2.247 88.338 0.038 GR-AN
-2.465 -44.225 39.295 0.988 GR-BS

$`30`
20-30 cm
2001 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
55.395 17.734 93.055 0.003 BS-AN
66.204 28.544 103.865 0.001 GR-AN
10.810 -25.726 47.346 0.742 GR-BS
# ECO
# lapply(split(sra.2001.bulk.df, sra.2001.bulk.df$lyr_bot), function(x) {
#   summary(lm(d14c ~ ECO, x))
# })
lapply(split(sra.2001.bulk.df, sra.2001.bulk.df$lyr_bot), function(x) {
  tukey.table.fx(x, "2001", "bulk", "ECO")
})
$`10`
0-10 cm
2001 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-71.522 -121.112 -21.931 0.004 rf-pp
-69.488 -117.597 -21.379 0.004 wf-pp
2.033 -47.557 51.623 0.994 wf-rf

$`20`
10-20 cm
2001 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-30.492 -77.815 16.831 0.260 rf-pp
-33.125 -79.035 12.785 0.190 wf-pp
-2.633 -49.956 44.690 0.989 wf-rf

$`30`
20-30 cm
2001 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-19.910 -71.507 31.686 0.605 rf-pp
-9.482 -59.538 40.574 0.884 wf-pp
10.429 -41.168 62.025 0.869 wf-rf
## inc
sra.2001.inc.df2 <- cbind(sra.19.01.inc.df[rep(1:nrow(sra.19.01.inc.df), 2), c("PM", "ECO", "lyr_bot")],
                          d14c = c(sra.19.01.inc.df$d14c_min, sra.19.01.inc.df$d14c_max))
save(sra.2001.inc.df2, file = "sra.2001.inc.df2.RData")
# PM
# lapply(split(sra.2001.inc.df2, sra.2001.inc.df2$lyr_bot), function(x) {
#   summary(lm(d14c ~ PM, x[x$d14c > -200, ]))
# })
lapply(split(sra.2001.inc.df2, sra.2001.inc.df2$lyr_bot), function(x) {
  tukey.table.fx(x, "2001", "inc", "PM")
})
$`10`
0-10 cm
2001 inc PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-8.178 -63.772 47.417 0.923 BS-AN
-3.095 -58.689 52.500 0.989 GR-AN
5.083 -50.512 60.678 0.969 GR-BS

$`20`
10-20 cm
2001 inc PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
23.755 -39.544 87.055 0.600 BS-AN
-44.282 -110.671 22.108 0.224 GR-AN
-68.037 -134.426 -1.648 0.044 GR-BS

$`30`
20-30 cm
2001 inc PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
41.442 -56.220 139.105 0.523 BS-AN
-61.708 -164.138 40.721 0.288 GR-AN
-103.151 -205.580 -0.721 0.048 GR-BS
# ECO
# lapply(split(sra.2001.inc.df2, sra.2001.inc.df2$lyr_bot), function(x) {
#   summary(lm(d14c ~ ECO, x[x$d14c > -200, ]))
# })
lapply(split(sra.2001.inc.df2, sra.2001.inc.df2$lyr_bot), function(x) {
  tukey.table.fx(x, "2001", "inc", "ECO")
})
$`10`
0-10 cm
2001 inc ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-54.988 -96.644 -13.332 0.010 rf-pp
-34.104 -75.760 7.552 0.118 wf-pp
20.883 -20.773 62.539 0.416 wf-rf

$`20`
10-20 cm
2001 inc ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-21.298 -100.627 58.032 0.766 rf-pp
6.974 -68.664 82.612 0.968 wf-pp
28.272 -51.058 107.601 0.629 wf-rf

$`30`
20-30 cm
2001 inc ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
8.875 -106.631 124.380 0.978 rf-pp
61.977 -48.153 172.107 0.333 wf-pp
53.103 -62.403 168.608 0.471 wf-rf
### 2019
## bulk
sra.2019.bulk.df <- bind_rows(sra.2019.ls)
# PM
# lapply(split(sra.2019.bulk.df, sra.2019.bulk.df$lyr_bot), function(x) {
#   if (nrow(x) == 27) summary(lm(d14c ~ PM, x[x$d14c > -200, ]))
# })
lapply(split(sra.2019.bulk.df, sra.2019.bulk.df$lyr_bot), function(x) {
  if (nrow(x) == 27) tukey.table.fx(x, "2019", "bulk", "PM")
})
$`10`
0-10 cm
2019 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-41.756 -76.898 -6.613 0.018 AN-GR
-9.522 -44.665 25.621 0.779 BS-GR
32.233 -2.909 67.376 0.077 BS-AN

$`20`
10-20 cm
2019 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-54.667 -106.932 -2.401 0.039 AN-GR
-16.478 -68.743 35.788 0.714 BS-GR
38.189 -14.077 90.455 0.183 BS-AN

$`30`
20-30 cm
2019 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-35.544 -84.884 13.796 0.191 AN-GR
-25.789 -75.129 23.551 0.406 BS-GR
9.756 -39.584 59.096 0.875 BS-AN

$`40`
30-40 cm
2019 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-31.478 -89.509 26.553 0.380 AN-GR
-19.733 -77.765 38.298 0.677 BS-GR
11.744 -46.287 69.776 0.869 BS-AN

$`50`
40-50 cm
2019 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-46.722 -107.064 13.620 0.151 AN-GR
-27.589 -87.931 32.753 0.498 BS-GR
19.133 -41.209 79.475 0.712 BS-AN

$`60`
50-60 cm
2019 bulk PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-9.289 -72.136 53.558 0.928 AN-GR
-7.044 -69.891 55.802 0.958 BS-GR
2.244 -60.602 65.091 0.996 BS-AN

$`70`
NULL

$`80`
NULL

$`90`
NULL
# ECO
# lapply(split(sra.2019.bulk.df, sra.2019.bulk.df$lyr_bot), function(x) {
#   if (nrow(x) == 27) summary(lm(d14c ~ ECO, x[x$d14c > -200, ]))
# })
lapply(split(sra.2019.bulk.df, sra.2019.bulk.df$lyr_bot), function(x) {
  if (nrow(x) == 27) tukey.table.fx(x, "2019", "bulk", "ECO")
})
$`10`
0-10 cm
2019 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-31.500 -65.613 2.613 0.074 wf-rf
14.222 -19.890 48.335 0.559 pp-rf
45.722 11.610 79.835 0.007 pp-wf

$`20`
10-20 cm
2019 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-35.556 -82.679 11.568 0.165 wf-rf
35.944 -11.179 83.068 0.159 pp-rf
71.500 24.376 118.624 0.002 pp-wf

$`30`
20-30 cm
2019 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-58.167 -94.986 -21.348 0.002 wf-rf
10.767 -26.052 47.586 0.748 pp-rf
68.933 32.114 105.752 0.000 pp-wf

$`40`
30-40 cm
2019 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-67.222 -107.865 -26.579 0.001 wf-rf
14.578 -26.065 55.221 0.648 pp-rf
81.800 41.157 122.443 0.000 pp-wf

$`50`
40-50 cm
2019 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-59.322 -105.522 -13.122 0.010 wf-rf
28.344 -17.855 74.544 0.294 pp-rf
87.667 41.467 133.866 0.000 pp-wf

$`60`
50-60 cm
2019 bulk ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-59.622 -114.836 -4.409 0.033 wf-rf
-27.478 -82.691 27.736 0.440 pp-rf
32.144 -23.069 87.358 0.330 pp-wf

$`70`
NULL

$`80`
NULL

$`90`
NULL
## inc
sra.2019.inc.df2 <- bind_rows(sra.2019.inc.ls)
save(sra.2019.inc.df2, file = "sra.2019.inc.df2.RData")
# PM
# lapply(split(sra.2019.inc.df2, sra.2019.inc.df2$lyr_bot), function(x) {
#   summary(lm(d14c ~ PM, x[x$d14c > -200, ]))
# })
lapply(split(sra.2019.inc.df2, sra.2019.inc.df2$lyr_bot), function(x) {
  tukey.table.fx(x, "2019", "inc", "PM")
})
$`10`
0-10 cm
2019 inc PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-3.917 -40.999 33.165 0.959 BS-AN
7.583 -29.499 44.665 0.857 GR-AN
11.500 -25.582 48.582 0.705 GR-BS

$`20`
10-20 cm
2019 inc PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-16.133 -76.600 44.333 0.768 BS-AN
-24.507 -87.925 38.911 0.582 GR-AN
-8.373 -71.791 55.045 0.937 GR-BS

$`30`
20-30 cm
2019 inc PM
PM.diff PM.lwr PM.upr PM.p.adj Pairs
-36.657 -86.659 13.346 0.170 BS-AN
-21.283 -68.959 26.392 0.490 GR-AN
15.373 -34.629 65.376 0.706 GR-BS
# ECO
# lapply(split(sra.2019.inc.df2, sra.2019.inc.df2$lyr_bot), function(x) {
#   summary(lm(d14c ~ ECO, x[x$d14c > -200, ]))
# })
lapply(split(sra.2019.inc.df2, sra.2019.inc.df2$lyr_bot), function(x) {
  tukey.table.fx(x, "2019", "inc", "ECO")
})
$`10`
0-10 cm
2019 inc ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-19.217 -54.416 15.983 0.357 rf-pp
-2.367 -37.566 32.833 0.983 wf-pp
16.850 -18.350 52.050 0.447 wf-rf

$`20`
10-20 cm
2019 inc ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-32.333 -87.289 22.623 0.303 rf-pp
-48.617 -101.015 3.782 0.071 wf-pp
-16.283 -71.239 38.673 0.724 wf-rf

$`30`
20-30 cm
2019 inc ECO
ECO.diff ECO.lwr ECO.upr ECO.p.adj Pairs
-14.457 -68.364 39.450 0.766 rf-pp
-21.933 -73.332 29.465 0.520 wf-pp
-7.477 -61.384 46.430 0.930 wf-rf
# compare 2001 and 2019
# bulk
sra.01.19.bulk.df <- data.frame(
  rbind(sra.2001.bulk.df, 
        sra.2019.bulk.df[, which(names(sra.2019.bulk.df) %in% names(sra.2001.bulk.df))]),
  year = as.factor(c(rep(2001, nrow(sra.2001.bulk.df)), rep(2019, nrow(sra.2019.bulk.df))))) %>%
  filter(lyr_bot < 31)
sra.01.19.bulk.ls <- split(sra.01.19.bulk.df, sra.01.19.bulk.df$PMeco)
lapply(sra.01.19.bulk.ls, function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$PMeco), " 2001 vs. 2019"), "bulk", "year")
  })
})
$ANpp
$ANpp$`10`
0-10 cm
ANpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-108.646 -161.123 -56.17 0.005 2019-2001

$ANpp$`20`
10-20 cm
ANpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-39.132 -121.572 43.307 0.258 2019-2001

$ANpp$`30`
20-30 cm
ANpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
19.62 -60.937 100.178 0.536 2019-2001


$ANrf
$ANrf$`10`
0-10 cm
ANrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-39.662 -97.2 17.876 0.116 2019-2001

$ANrf$`20`
10-20 cm
ANrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-23.451 -62.372 15.471 0.151 2019-2001

$ANrf$`30`
20-30 cm
ANrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
3.602 -44.344 51.548 0.826 2019-2001


$ANwf
$ANwf$`10`
0-10 cm
ANwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
2.096 -54.912 59.104 0.924 2019-2001

$ANwf$`20`
10-20 cm
ANwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
7.388 -22.778 37.553 0.534 2019-2001

$ANwf$`30`
20-30 cm
ANwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
5.913 -13.918 25.744 0.454 2019-2001


$BSpp
$BSpp$`10`
0-10 cm
BSpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-34.257 -85.03 16.516 0.134 2019-2001

$BSpp$`20`
10-20 cm
BSpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-4.4 -43.862 35.063 0.772 2019-2001

$BSpp$`30`
20-30 cm
BSpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-16.345 -82.688 49.998 0.532 2019-2001


$BSrf
$BSrf$`10`
0-10 cm
BSrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
14.282 -37.804 66.367 0.489 2019-2001

$BSrf$`20`
10-20 cm
BSrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-8.656 -67.505 50.193 0.704 2019-2001

$BSrf$`30`
20-30 cm
BSrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
27.438 -46.279 101.155 0.36 2019-2001


$BSwf
$BSwf$`10`
0-10 cm
BSwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-36.239 -83.527 11.049 0.1 2019-2001

$BSwf$`20`
10-20 cm
BSwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-64.075 -115.738 -12.412 0.026 2019-2001

$BSwf$`30`
20-30 cm
BSwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-113.138 -153.195 -73.08 0.001 2019-2001


$GRpp
$GRpp$`10`
0-10 cm
GRpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-51.541 -120.524 17.443 0.107 2019-2001

$GRpp$`20`
10-20 cm
GRpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
36.232 -32.197 104.661 0.215 2019-2001

$GRpp$`30`
20-30 cm
GRpp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
2.096 -49.559 53.75 0.916 2019-2001


$GRrf
$GRrf$`10`
0-10 cm
GRrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-4.878 -56.92 47.164 0.808 2019-2001

$GRrf$`20`
10-20 cm
GRrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
9.203 -13.93 32.335 0.331 2019-2001

$GRrf$`30`
20-30 cm
GRrf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
6.556 -18.916 32.027 0.514 2019-2001


$GRwf
$GRwf$`10`
0-10 cm
GRwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-89.004 -114.543 -63.464 0.001 2019-2001

$GRwf$`20`
10-20 cm
GRwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-65.737 -141.304 9.829 0.073 2019-2001

$GRwf$`30`
20-30 cm
GRwf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-65.759 -140.676 9.159 0.071 2019-2001
# by PM
lapply(split(sra.01.19.bulk.df, sra.01.19.bulk.df$PM), function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$PM), " 2001 vs. 2019"), "bulk", "year")
  })
})
$AN
$AN$`10`
0-10 cm
AN 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-47.196 -96.931 2.54 0.061 2019-2001

$AN$`20`
10-20 cm
AN 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-16.142 -50.009 17.725 0.326 2019-2001

$AN$`30`
20-30 cm
AN 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
11.624 -21.346 44.594 0.464 2019-2001


$BS
$BS$`10`
0-10 cm
BS 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-18.738 -51.619 14.142 0.245 2019-2001

$BS$`20`
10-20 cm
BS 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-25.71 -57.388 5.967 0.105 2019-2001

$BS$`30`
20-30 cm
BS 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-34.015 -76.337 8.307 0.108 2019-2001


$GR
$GR$`10`
0-10 cm
GR 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-48.474 -89.466 -7.482 0.023 2019-2001

$GR$`20`
10-20 cm
GR 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-6.768 -59.155 45.62 0.788 2019-2001

$GR$`30`
20-30 cm
GR 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-19.036 -54.623 16.551 0.274 2019-2001
# by ECO
lapply(split(sra.01.19.bulk.df, sra.01.19.bulk.df$ECO), function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$ECO), " 2001 vs. 2019"), "bulk", "year")
  })
})
$pp
$pp$`10`
0-10 cm
pp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-64.815 -100.543 -29.086 0.001 2019-2001

$pp$`20`
10-20 cm
pp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-2.433 -50.649 45.783 0.916 2019-2001

$pp$`30`
20-30 cm
pp 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
1.79 -36.185 39.766 0.922 2019-2001


$rf
$rf$`10`
0-10 cm
rf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-7.515 -30.699 15.668 0.5 2019-2001

$rf$`20`
10-20 cm
rf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-7.886 -35.016 19.245 0.545 2019-2001

$rf$`30`
20-30 cm
rf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
10.934 -15.022 36.89 0.383 2019-2001


$wf
$wf$`10`
0-10 cm
wf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-41.049 -84.594 2.497 0.063 2019-2001

$wf$`20`
10-20 cm
wf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-40.808 -80.859 -0.757 0.046 2019-2001

$wf$`30`
20-30 cm
wf 2001 vs. 2019 bulk year
year.diff year.lwr year.upr year.p.adj Pairs
-57.661 -102.561 -12.761 0.015 2019-2001
# inc
sra.01.19.inc.df <- data.frame(
  d14c = c(sra.19.01.inc[ , "d14c_min"],
           sra.19.01.inc[ , "d14c_max"]),
  sra.19.01.inc[ , c("PMeco", "lyr_bot", "PM", "ECO", "year")]) %>%
  mutate(year = as.factor(year))
sra.01.19.inc.ls <- split(sra.01.19.inc.df, sra.01.19.inc.df$PMeco)
lapply(sra.01.19.inc.ls, function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$PMeco), " 2001 vs. 2019"), "inc", "year")
  })
})
NaNs producedNaNs producedNaNs produced
$ANpp
$ANpp$`10`
0-10 cm
ANpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-112.068 -168.378 -55.758 0.013 2019-2001

$ANpp$`20`
10-20 cm
ANpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-41.107 -126.069 43.856 0.173 2019-2001

$ANpp$`30`
20-30 cm
ANpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
26.321 -162.239 214.88 0.609 2019-2001


$ANrf
$ANrf$`10`
0-10 cm
ANrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-53.528 -70.509 -36.547 0.005 2019-2001

$ANrf$`20`
10-20 cm
ANrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-20.65 -71.163 29.863 0.221 2019-2001

$ANrf$`30`
20-30 cm
ANrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
20.616 -7.758 48.989 0.089 2019-2001


$ANwf
$ANwf$`10`
0-10 cm
ANwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-25.061 -123.937 73.815 0.39 2019-2001

$ANwf$`20`
10-20 cm
ANwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-18.73 -25.823 -11.636 0.007 2019-2001

$ANwf$`30`
20-30 cm
ANwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-26.671 -63.212 9.87 0.088 2019-2001


$BSpp
$BSpp$`10`
0-10 cm
BSpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-41.7 -75.535 -7.865 0.034 2019-2001

$BSpp$`20`
10-20 cm
BSpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-20.628 -38.25 -3.006 0.037 2019-2001

$BSpp$`30`
20-30 cm
BSpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
9.436 -108.88 127.751 0.764 2019-2001


$BSrf
$BSrf$`10`
0-10 cm
BSrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-69.149 -82.933 -55.365 0.002 2019-2001

$BSrf$`20`
10-20 cm
BSrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-72.152 -162.05 17.746 0.075 2019-2001

$BSrf$`30`
20-30 cm
BSrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-61.935 -70.107 -53.763 0 2019-2001


$BSwf
$BSwf$`10`
0-10 cm
BSwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-67.026 -82.623 -51.43 0.002 2019-2001

$BSwf$`20`
10-20 cm
BSwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-107.373 -168.048 -46.697 0.017 2019-2001

$BSwf$`30`
20-30 cm
BSwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-139.763 -241.947 -37.579 0.028 2019-2001


$GRpp
$GRpp$`10`
0-10 cm
GRpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-89.459 -114.521 -64.397 0.004 2019-2001

$GRpp$`20`
10-20 cm
GRpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
25.06 -77.741 127.86 0.405 2019-2001

$GRpp$`30`
20-30 cm
GRpp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
49.63 -228.753 328.013 0.524 2019-2001


$GRrf
$GRrf$`10`
0-10 cm
GRrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-13.237 -147.681 121.206 0.713 2019-2001

$GRrf$`20`
10-20 cm
GRrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
69.3 NaN NaN NaN 2019-2001

$GRrf$`30`
20-30 cm
GRrf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
166.594 NaN NaN NaN 2019-2001


$GRwf
$GRwf$`10`
0-10 cm
GRwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-55.927 -101.954 -9.899 0.035 2019-2001

$GRwf$`20`
10-20 cm
GRwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-77.344 -162.104 7.416 0.059 2019-2001

$GRwf$`30`
20-30 cm
GRwf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
0.089 -36.296 36.474 0.993 2019-2001
lapply(split(sra.01.19.inc.df, sra.01.19.inc.df$PM), function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$PM), " 2001 vs. 2019"), "inc", "year")
  })
})
$AN
$AN$`10`
0-10 cm
AN 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-63.553 -94.114 -32.991 0.001 2019-2001

$AN$`20`
10-20 cm
AN 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-26.829 -48.81 -4.847 0.022 2019-2001

$AN$`30`
20-30 cm
AN 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
6.755 -37.035 50.545 0.738 2019-2001


$BS
$BS$`10`
0-10 cm
BS 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-59.292 -79.352 -39.232 0 2019-2001

$BS$`20`
10-20 cm
BS 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-66.718 -102.636 -30.799 0.002 2019-2001

$BS$`30`
20-30 cm
BS 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-64.087 -123.626 -4.549 0.037 2019-2001


$GR
$GR$`10`
0-10 cm
GR 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-52.874 -112.814 7.065 0.078 2019-2001

$GR$`20`
10-20 cm
GR 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-7.054 -105.253 91.146 0.873 2019-2001

$GR$`30`
20-30 cm
GR 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
47.18 -45.261 139.621 0.278 2019-2001
lapply(split(sra.01.19.inc.df, sra.01.19.inc.df$ECO), function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$ECO), " 2001 vs. 2019"), "inc", "year")
  })
})
$pp
$pp$`10`
0-10 cm
pp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-81.076 -110.49 -51.662 0 2019-2001

$pp$`20`
10-20 cm
pp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-12.225 -41.041 16.591 0.367 2019-2001

$pp$`30`
20-30 cm
pp 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
28.462 -20.711 77.636 0.226 2019-2001


$rf
$rf$`10`
0-10 cm
rf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-45.305 -84.505 -6.104 0.028 2019-2001

$rf$`20`
10-20 cm
rf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-23.261 -119.616 73.094 0.593 2019-2001

$rf$`30`
20-30 cm
rf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
9.738 -101.496 120.971 0.847 2019-2001


$wf
$wf$`10`
0-10 cm
wf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-49.338 -79.02 -19.657 0.004 2019-2001

$wf$`20`
10-20 cm
wf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-67.816 -112.331 -23.3 0.007 2019-2001

$wf$`30`
20-30 cm
wf 2001 vs. 2019 inc year
year.diff year.lwr year.upr year.p.adj Pairs
-55.448 -108.861 -2.035 0.043 2019-2001
NANA
sra.01.19.min.reps <- left_join(
  merge(sra.01.19.bulk.df, sra.01.19.inc.df,
        by = c("PMeco", "PM", "ECO", "year", "lyr_bot"),
        suffixes = c("_blk", "_inc")),
  ras18.sp.df[ , c("Al_ox", "Al_py", "Fe_dc", "Fe_ox", "PM", "ECO", "lyr_bot")],
  by = c("PM", "ECO", "lyr_bot")) %>%
  mutate(Year = as.numeric(as.character(year)))

summary(lm(d14c_blk ~ Al_ox + lyr_bot + year, sra.01.19.min.reps))

Call:
lm(formula = d14c_blk ~ Al_ox + lyr_bot + year, data = sra.01.19.min.reps)

Residuals:
    Min      1Q  Median      3Q     Max 
-89.532 -27.114  -0.197  26.055 115.938 

Coefficients:
            Estimate Std. Error t value     Pr(>|t|)    
(Intercept) 108.5757     6.0801  17.858      < 2e-16 ***
Al_ox        -1.9945     0.1687 -11.822      < 2e-16 ***
lyr_bot      -3.0592     0.2487 -12.300      < 2e-16 ***
year2019    -22.4127     4.0521  -5.531 0.0000000671 ***
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

Residual standard error: 36.11 on 314 degrees of freedom
Multiple R-squared:  0.527, Adjusted R-squared:  0.5225 
F-statistic: 116.6 on 3 and 314 DF,  p-value: < 2.2e-16
sra.01.19.min.reps %>%
  mutate(eco = ifelse(ECO == "rf", "cold", ifelse(ECO == "wf", "cool", "warm"))) %>%
  mutate(ecoYear = paste0(eco, " (", year, ")")) %>%
  # filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  ggplot(., aes(Al_ox, d14c_blk)) +
  geom_point(aes(color = PM, shape = ecoYear), size = 3) +
  scale_color_manual(name = "Parent material",
                     values = c("AN" = andesite,
                                "BS" = basalt,
                                "GR" = granite),
                     labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite")) + 
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  facet_wrap(vars(lyr_bot)) +
  ylab(expression('Bulk '*Delta*''^14*'C (‰)')) +
  xlab(expression('Oxalate extractable Al (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# color palettes for ECO & PM
warm <- "#BF812D"
cool <- "#80CDC1"
cold <- "#01665E"
granite <- "#9daba9"
andesite <- "#382dbf"
basalt <- "#bf382d"

# plot fx
boxplot.fx <- function(df, var, year, type, topsoil = FALSE, subsoil = FALSE) {
  atm <- ifelse(year == "2001", atm.d14.2001, atm.d14.2019)
  if (type == "inc") {
    df <- df[df$d14c > -200, ]
    ylim <- c(-65, 165)
  } else {
    if (topsoil) {
      df <- df[df$lyr_bot < 31, ]
      ylim <- c(-120, 165)
      }
    if (subsoil) {
      df <- df[df$lyr_bot > 31, ]
      ylim <- c(-270, 65)
    }
  }
  if (var == "PM") {
    df %>%
      mutate(pm = factor(ifelse(PM == "GR", "granite",
                                ifelse(PM == "AN", "andesite", "basalt")),
                         levels = c("granite", "andesite", "basalt"))) %>%
      group_by(pm, lyr_bot) %>%
      ggplot(., aes(pm, d14c)) +
      geom_hline(yintercept = atm, linetype = "dotted", alpha = 0.3) +
      geom_hline(yintercept = 0) +
      geom_boxplot(aes(color = pm), lwd = 1) +
      scale_color_manual(values = c("andesite" = andesite,
                                    "basalt" = basalt,
                                    "granite" = granite),
                         guide = "none") +
      scale_y_continuous(limits = ylim) +
      facet_grid(cols = vars(lyr_bot)) +
      ylab(expression(Delta*''^14*'C (‰)')) +
      ggtitle(paste(year, type)) +
      theme_bw() +
      theme(panel.grid = element_blank(),
            text = element_text(size = 14))
  } else {
    df %>%
      mutate(eco = factor(ifelse(ECO == "pp", "warm",
                                 ifelse(ECO == "wf", "cool", "cold")),
                          levels = c("warm", "cool", "cold"))) %>%
      group_by(eco, lyr_bot) %>%
      ggplot(., aes(eco, d14c)) +
      geom_hline(yintercept = atm, linetype = "dotted", alpha = 0.3) +
      geom_hline(yintercept = 0) +
      geom_boxplot(aes(color = eco), lwd = 1) +
      scale_color_manual(values = c("warm" = warm,
                                    "cool" = cool,
                                    "cold" = cold),
                         guide = "none") +
      scale_y_continuous(limits = ylim) +
      facet_grid(cols = vars(lyr_bot)) +
      ylab(expression(Delta*''^14*'C (‰)')) +
      ggtitle(paste(year, type)) +
      theme_bw() +
      theme(panel.grid = element_blank(),
            text = element_text(size = 14))
  }
}

# bulk
boxplot.fx(sra.2001.bulk.df, "PM", "2001", "bulk", topsoil = TRUE)

boxplot.fx(sra.2019.bulk.df, "PM", "2019", "bulk", topsoil = TRUE)

boxplot.fx(sra.2001.bulk.df, "ECO", "2001", "bulk", topsoil = TRUE)

boxplot.fx(sra.2019.bulk.df, "ECO", "2019", "bulk", topsoil = TRUE)

boxplot.fx(sra.2019.bulk.df, "ECO", "2019", "bulk", subsoil = TRUE)

# inc
boxplot.fx(sra.2001.inc.df2, "PM", "2001", "inc")

boxplot.fx(sra.2019.inc.df2, "PM", "2019", "inc")

boxplot.fx(sra.2001.inc.df2, "ECO", "2001", "inc")

boxplot.fx(sra.2019.inc.df2, "ECO", "2019", "inc")

# data, unsummarized
sra.ts.all.raw <- rbind(
  sra.2001.bulk.df[ , names(sra.2001.bulk.df) %in% names(sra.2001.inc.df2)],
  sra.2019.bulk.df[ , names(sra.2019.bulk.df) %in% names(sra.2001.inc.df2)],
  sra.2001.inc.df2,
  sra.2019.inc.df2[ , names(sra.2019.inc.df2) %in% names(sra.2001.inc.df2)]) %>%
  mutate(eco = factor(ifelse(ECO == "pp", "warm",
                             ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = factor(ifelse(PM == "GR", "granite",
                                ifelse(PM == "AN", "andesite", "basalt")),
                         levels = c("granite", "andesite", "basalt")),
         Type = c(rep("bulk", length = nrow(sra.2001.bulk.df)),
                  rep("bulk", length = nrow(sra.2019.bulk.df)),
                  rep("inc", length = nrow(sra.2001.inc.df2)),
                  rep("inc", length = nrow(sra.2019.inc.df2))),
         year = c(rep(2001, length = nrow(sra.2001.bulk.df)),
                  rep(2019, length = nrow(sra.2019.bulk.df)),
                  rep(2001, length = nrow(sra.2001.inc.df2)),
                  rep(2019, length = nrow(sra.2019.inc.df2))))

# plot fx
ts.groupPlot.fx <- function(df, x, y) {
  quo_x <- sym(x)
  quo_y <- sym(y)
  if (x == "pm") {
    var.name <- "Parent material"
    var.values <- c("andesite" = andesite,
                    "basalt" = basalt,
                    "granite" = granite) 
  } else {
    var.name <- "Climate"
    var.values <-  c("warm" = warm,
                     "cool" = cool,
                     "cold" = cold)
  }
  plot.df <- df %>%
    filter(d14c > -200) %>%
    filter(lyr_bot < 31) %>%
    group_by(!! quo_x, lyr_bot, Type, year) %>%
    summarize(across(d14c, list(mean = mean, sd = sd)))
  if (y == "dd14c") {
    plot.df <- plot.df %>%
      mutate(atm = ifelse(year == 2001, atm.d14.2001, atm.d14.2019),
             dd14c = d14c_mean - atm,
             u = d14c_mean + d14c_sd - atm,
             l = d14c_mean - d14c_sd - atm)
    atm.df <- atm.14c
    atm.df$d14c <- 0
    ylab <- expression(Delta*Delta*''^14*'C (‰)') 
    } else {
      plot.df <- plot.df %>%
        mutate(u = d14c_mean + d14c_sd,
               l = d14c_mean - d14c_sd)
      atm.df <- atm.14c
      ylab <- expression(Delta*''^14*'C (‰)') 
    }
    ggplot(plot.df, aes(year, !! quo_y)) +
    geom_path(data = atm.df, aes(year, d14c)) +
    geom_path(aes(color = !! quo_x, linetype = Type), alpha = .5, lwd = 1) +
    geom_point(aes(color = !! quo_x), 
               size = 3, position = position_dodge(width = 1)) +
    geom_errorbar(
      aes(ymin = l,
          ymax = u,
          color = !! quo_x,
          alpha = Type),
      width = 1,
      position = position_dodge(width = 1)) +
    scale_color_manual(name = var.name,
                       values = var.values) +
    scale_fill_manual(name = "±SD",
                      values = var.values) +
    scale_alpha_manual(values = c("bulk" = 1,
                                  "inc" = .5)) +
    facet_grid(cols = vars(lyr_bot)) +
    ylab(ylab) +
    xlab("Year") +
    theme_bw() +
    theme(panel.grid = element_blank())
}
# plot
ts.groupPlot.fx(sra.ts.all.raw, "pm", "dd14c")

ts.groupPlot.fx(sra.ts.all.raw, "eco", "dd14c")

ts.groupPlot.fx(sra.ts.all.raw, "pm", "d14c_mean")

ts.groupPlot.fx(sra.ts.all.raw, "eco", "d14c_mean")

Initial modeling

The goal of this modeling exercise is to see how parent material and climate/ecosystem affect estimates of soil carbon ages and transit times. Bulk soil 14C observations from 2001, 2009, and 2019 will be used to constrain the carbon models, as well as observations of 14C-CO2 from laboratory soil incubations of soils collected in 2001 and 2019. Previous work has indicated that the carbon stocks at these sites is likely at equilibrium, so we will apply the steady-state assumption to the modeling.

Two-pool models

One pool models have been shown repeatedly to be inadequate for describing soil carbon dynamics. However, as simple models are easier to constrain, we will start with a two-pool parallel and two-series models, as these are the simplest model system beyond the single pool approach.

The two-pool parallel model requires the following parameters: * decomposition constants for each pool (k1, k2) * input partitioning coefficient (\(\gamma\)) * steady-state carbon stocks (C) * inputs (I) * initial values of 14C 1 The two-pool series model requires the following parameters: * decomposition constants for each pool (k1, k2) * transfer coefficient (\(\alpha\)) * steady-state carbon stocks (C) * inputs (I) * initial values of 14C

Decomposition rates (k) are related to the amount of 14C in a pre-bomb system (fraction modern, F) at steady-state by the following equations (cf. Schuur, Druffle, and Trumbore, 2016): >Eq. 1

\[F = \frac{k}{k + \lambda}\] >Eq. 2

\[k = \frac{\lambda \cdot F}{1 - F}\] >where \(\lambda\) is the radioactive decay constant (1/8267).

As the decomposition rates will vary, the initial 14C content can be determined dynamically with equation 1.

Carbon stocks are known, while inputs will be estimated and are related to the steady-state conditions by the following equation: >Eq. 3

\[I = (k_{1} \cdot C_{1}) + (k_{2} \cdot C_{2})\] >where C1 and C2 are the carbon stocks of the two model pools.

Both stocks and inputs can be scaled to the known value of the total carbon pool once the steady-state parameters (k1, k2, and \(\gamma\) or \(\alpha\)) have been determined. Pool sizes are a function of the inputs and input partitioning coefficient at steady-state.

A Monte-Carlo Markov chain approach will be used for parameter estimation in combination with an initial optimization algorithm to determine the best set of initial parameters.

Workflow

Initial model fitting was performed for both model structures using generous parameter ranges [0, 1] for all three parameters (k1, k2, \(\gamma\) or \(\alpha\)). The initial parameter set was found by fitting the models by eye, followed by optimization with the function “modFit” (R package FME), using the Nelder-Mead algorithm. The best set of parameters found by modFit was then used as the input to a Monte Carlo Markov Chain (MCMC), using the function “modMCMC” (R package FME). The number of iterations for the MCMC optimization was set at 5000 intially, with delayed rejection employed to increase efficiency.

The sum of the mean squared error for the best parameter set was slightly lower for the parallel structure than for the series structure. Additionally, the overall mean error of the residuals was also lower for the parallel structure, moderately so for the bulk C observations but substantially so for the respiration observations (in andesite and granite soils in particular).

However, these initial fits yielded unrealistic parameter estimates for multiple sites, particularly at the lower depths. Additionally, the modFit output showed very high correlation between the parameters for both model structures (slightly higher for the two-pool series model).

# k from fraction modern
k <- function (Fm) {
  (Fm * lambda)/(1 - Fm)
}

# d14C from fraction modern 
fm_14c <- function (fm, date) {
  (fm * exp(lambda * (1950 - date)) - 1) * 1000
}

# pre-bomb fraction modern from k (steady-state assumed)
fm <- function (k){
  k/(k + lambda)
}
# index of years for which bulk/resp 14C are known
year.ix <- c(which(Datm$Date == 2001.5),
             which(Datm$Date == 2009.5),
             which(Datm$Date == 2019.5))

# function for saving constraint data in a dataframe for plotting in ggplot'
con.df.fx <- function(PMeco_depth) {
  bulk.df <- obs.bulk.14c[[PMeco_depth]]
  resp.df <- obs.resp.14c[[PMeco_depth]]
  return(
    con.df <- data.frame(pool = c(rep("bulk C", nrow(bulk.df)), rep("respiration", nrow(resp.df))),
                         d14c = c(bulk.df$bulkC, resp.df$resp),
                         Year = c(bulk.df$time, resp.df$time)))
}

# plot function
C14.2p.plot.fx <- function(plot.df, con.df, mod, PMeco_depth) {
  plot.df %>%
  filter(pool == "bulk C" | pool == "respiration" | pool == "atm") %>%
  ggplot(., aes(years, d14C, color = pool)) +
  geom_path() +
  geom_point(data = con.df, aes(Year, d14c, color = pool), size = 3) +
  scale_color_manual(
    name = "Pool",
    values = c("atm" = 8,
               "bulk C" = "black",
               "fast" = "#D81B60",
               "slow" = "#1E88E5",
               "respiration" = "#FFC107")) +
  scale_x_continuous(limits = c(1950, 2022)) +
  ggtitle(paste(PMeco_depth, mod)) +
  xlab("Year") +
  ylab(expression(''*Delta*''^14*'C (‰)')) +
  theme_bw() +
  theme(panel.grid = element_blank())
}
C14.1p.plot.fx <- function(plot.df, con.df, mod, PMeco_depth) {
  ggplot(plot.df, aes(years, d14C, color = pool)) +
  geom_path() +
  geom_point(data = con.df, aes(Year, d14c, color = pool), size = 3) +
  scale_color_manual(
    name = "Pool",
    values = c("atm" = 8,
               "bulk C" = "black",
               "respiration" = "#FFC107")) +
  scale_x_continuous(limits = c(1950, 2022)) +
  ggtitle(paste(PMeco_depth, " 1p bulk + 1p resp")) +
  xlab("Year") +
  ylab(expression(''*Delta*''^14*'C (‰)')) +
  theme_bw() +
  theme(panel.grid = element_blank())
}

# set up model function for optimization
# NOTE: par[3] for 2ps model changed to proportion transferred (no longer = a21)
# therefore, a21 = par[3] * par[1]
modFun_2p <- function(pars, In, lag = 0, pass = TRUE, out = "modFit", mod) {
 
  # intial 14C
  F0_Delta14C <- unlist(lapply(pars[1:2], function(x) Delta14C_from_AbsoluteFractionModern(fm(x))))
  
  # model matrix
  A <- -diag(pars[1:2])
  if (mod == "2ps") {
    a21 <- pars[3] * pars[1]
    A[2, 1] <- a21
  }
    
  # steady-state C stocks
  if (mod == "2pp") {
    ss.cstock <- (-1 * solve(A) %*% c(In * pars[3], In * (1 - pars[3])))
  } else {
    ss.cstock <- (-1 * solve(A) %*% c(In, 0))
  }
  
  # time index
  ix.t <- c((lag + 1):nrow(Datm))
  
  # model
  if (mod == "2pp") {
    mod <- TwopParallelModel14(t = Datm$Date[ix.t],
                               ks = pars[1:2],
                               C0 = c(ss.cstock[1], ss.cstock[2]),
                               F0_Delta14C = F0_Delta14C,
                               In = In,
                               gam = pars[3],
                               inputFc = Datm,
                               lag = lag,
                               pass = pass)
  } else {
    mod <- TwopSeriesModel14(t = Datm$Date[ix.t],
                             ks = pars[1:2],
                             C0 = c(ss.cstock[1], ss.cstock[2]),
                             F0_Delta14C = F0_Delta14C,
                             In = In,
                             a21 = a21,
                             inputFc = Datm,
                             lag = lag,
                             pass = pass)
  }
  
  # get mod values
  C14m <- getF14C(mod)
  C14p <- getF14(mod)
  C14r <- getF14R(mod)
  Ctot <- getC(mod)
  
  if(out == "modFit") {
    # dataframe for modFit fx
    return(data.frame(
      time = Datm$Date[ix.t],
      bulkC = C14m, 
      resp = C14r,
      cStock = rowSums(Ctot)))
  } else {
    # data frame for plotting
    return(data.frame(
      years = rep(Datm$Date[ix.t], 5),
      d14C = c(C14p[,1], 
               C14p[,2], 
               C14m,
               C14r,
               Datm$NHc14[ix.t]),
      pool = rep(c("fast", "slow", "bulk C", "respiration", "atm"), each = nrow(C14p))))
  }
}

# 1p modFun
modFun_1p <- function(pars, In, lag = 0, out = "modFit", mod, pass = TRUE) {
 
  # intial 14C
  F0_Delta14C <- Delta14C_from_AbsoluteFractionModern(fm(pars))
  
  # steady-state C stocks
  ss.cstock <- In/pars
  
  # time index
  ix.t <- c((lag + 1):nrow(Datm))
  
  # model
  mod <- suppressWarnings(
    # warnings suppressed due to the "Fc" warning
    OnepModel14(t = Datm$Date[ix.t],
                     k = pars,
                     C0 = ss.cstock,
                     F0_Delta14C = F0_Delta14C,
                     In = In,
                     inputFc = Datm,
                     lag = lag,
                     pass = pass)
  )
  
  # get mod values
  C14m <- getF14C(mod)
  Ctot <- getC(mod)
  
  if(out == "modFit") {
    # dataframe for modFit fx
    return(data.frame(
      time = Datm$Date[ix.t],
      bulkC = C14m,
      cStock = Ctot))
  } else {
    # data frame for plotting
    return(data.frame(
      years = rep(Datm$Date[ix.t], 1),
      d14C = c(C14m,
               Datm$NHc14[ix.t]),
      pool = rep(c("bulk C", "atm"), each = length(C14m))))
  }
}

# function for trial and error approach to finding initial parameter set
par.fx <- function(pars, In, lag = 0, out = "plot.df", verbose = TRUE, mod, pass = FALSE) {
  
  # model matrix
  A <- -diag(pars[1:2])
  if (mod == "2ps") {
    a21 <- pars[3] * pars[1]
    A[2, 1] <- a21
    # steady-state stocks
    ss.cstock <- round((-1 * solve(A) %*% c(In, 0)), 1)
  } else if (mod == "2pp") {
    # steady-state stocks
    ss.cstock <- round((-1 * solve(A) %*% c(In * pars[3], In * (1 - pars[3]))), 1)
  } else {
    ss.cstock <- In/pars
  }
  
  cstock.sum <- ifelse(length(ss.cstock) == 1, ss.cstock, colSums(ss.cstock))
  
  # print site and steady-state stocks
  if (verbose) {
    cat(paste0(PMeco_depth, "\n"))
    if (mod == "2ps" | mod == "2pp") {
      cat(paste0(ss.cstock[1], " (fast pool)\n", ss.cstock[2], " (slow pool)\n"))
      cat(paste0("slow pool: ", round(ss.cstock[2] / cstock.sum * 100, 0), "%\n")) 
    }
    cat(round(cstock.sum, 1), " (modeled stocks)\n")
    cat(round(csoc.19.0_30[[PMeco_depth]][ , "lyr_soc"], 1), " (measured stocks)\n") 
  }
  if (mod == "1p") {
    return(modFun_1p(pars = pars, In = In, lag = lag, out = out, mod = "1p", pass = pass))
  }
  if (mod == "2pp") {
   return(modFun_2p(pars = pars, In = In, lag = lag, out = out, mod = "2pp", pass = pass)) 
  } else if (mod == "2ps") {
    return(modFun_2p(pars = pars, In = In, lag = lag, out = out, mod = "2ps", pass = pass)) 
  }
}
## adjust inputs to match stocks
# function for calculating steady-state SOC stocks
soc.fx <- function(modStr, pars, In, out = "pools") {
  if (modStr == "2pp") {
    cmat <- -1 * solve(-diag(pars[1:2])) %*% c(In * pars[3], In * (1 - pars[3]))
  } else {
    A <- -diag(pars[1:2])
    A[2, 1] <- pars[3] # note that a21 defined as pct transfer * k1
    cmat <- -1 * solve(A) %*% c(In, 0) # In is total input into the system
  }
  if (out == "pools") {
    return(cmat)
  } else {
    return(colSums(cmat))
  }
}

in.fit.fx <- function(modStr, pars, initialIn, SOC) {
  # sequence of possible input values
  if  (SOC < soc.fx(modStr, pars, initialIn, out = "sum")) {
    ins <- seq(.01, 
               initialIn, 
               .01)
    } else {
      ins <- seq(initialIn, 
                 SOC, 
                 .01)
    }
  # modeled stocks
  soc_mod <- lapply(seq_along(ins), function(j) {
    soc.fx(modStr, pars, ins[j], out = "sum")
  })
  ix <- which.min(abs(unlist(soc_mod) - SOC))
  return(ins[ix])
}

# load initial parameter set
load("../data/derived/modFit_pars/pars.i.2pp_2021-03-30.Rdata")
load("../data/derived/modFit_pars/pars.i.2ps_2020-11-16.Rdata")

## inputs for initial par set and measured stocks
# 2pp
in.meas.2pp <- lapply(seq_along(pars.i.2pp[ix.10]), function(i) {
  PMeco_depth <- names(pars.i.2pp[ix.10])[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.i.2pp[ix.10][[i]], in.i[ix.10][[i]], SOC))
})
names(in.meas.2pp) <- names(pars.i.2pp[ix.10])
# 2ps
in.meas.2ps <- lapply(seq_along(pars.i.2ps[ix.10]), function(i) {
  PMeco_depth <- names(pars.i.2ps[ix.10])[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.i.2ps[ix.10][[i]], in.i[ix.10][[i]], SOC))
})
names(in.meas.2ps) <- names(pars.i.2ps[ix.10])

# Flux estimated from Goulden et al. 2012; Tang et al. 2005; Wang et al. 2000; Gaudinski 2000
# fluxes by elevation from GPP reported in Goulden et al. Fig. 5 and approximated
# Rh percentage from Tang et al. 2005 = 0.44 (ann. mean Blodgett); cf. 0.48 Harvard Forest
# A horizon est. 0.55 from Gaudinski
# assuming A = 0-30, assume 0-10 = 50%, 10-20 = 30%, 20-30 = 20% of total A production 
hznA.Rh.kgm2 <- 0.44 * 0.55 * 10^-3
gpp.ls <- c(1800, 1600, 1400)
in.frc.ls <- c(0.5, 0.3, 0.2)

# fx for calculating inputs
in.flx.fx <- function(PMeco_depth) {
  gpp <- ifelse(grepl("pp", PMeco_depth), gpp.ls[1], ifelse(grepl("wf", PMeco_depth), gpp.ls[2], gpp.ls[3]))
  in.frc <- ifelse(grepl("0-10", PMeco_depth), in.frc.ls[1], ifelse(grepl("10-20", PMeco_depth), in.frc.ls[2], in.frc.ls[3]))
  return(gpp * in.frc * hznA.Rh.kgm2)
}

# input list
in.est <- lapply(seq_along(pars.i.2pp), function(i) {
  PMeco_depth <- names(pars.i.2pp)[i]
  return(in.flx.fx(PMeco_depth))
})
names(in.est) <- names(pars.i.2pp)

Parameter optimization

Optimizing the parameter set requires imposing costs and optionally constraining the allowable range of values for each parameter. Given that we only have data for three time points, this is a relatively sparse data set for constraining these models. Accordingly, the optimization procedure will benefit from a priori constraints of the allowable parameter ranges. For example, since we assume that the system cannot be adequately modeled as a single homogenous reservoir, we will ensure that the optimization procedure cannot collapse the two-pool system into a single pool. This can be mitigated in the two-pool parallel optimization by constraining \(\gamma\) (i.e. the percentage of the inputs entering the fast pool) to a range of 50% to 95%. Similarly, for the two-pool series model structure we can constrain the range of the transfer coefficient to be between 0.0 and 0.1, ensuring that some carbon remains in the fast cycling pool.

Additionally, to enforce a relatively fast cycling pool and relatively slower cycling pool, we will loosely constrain the intrinsic decomposition rates as well (both model structures):

k1: [0.02, 1.00] (50 to 1 year) k2: [0.0001, 0.02] (10,000 to 50 years)

Finally, the models will be run to enforce steady-state, i.e. with unvarying carbon stocks. The amount of carbon observed in the system will be used in the cost function in addition to the radiocarbon observations made in 2001, 2009, and 2019. The inputs will be estimated from net ecosystem exchange (NEE) data measured at nearby eddy covariance sites: Blodgett experimental forest (AmeriFlux), Lower Teakettle (NEON), and Soaproot Saddle (NEON). Alternatively, using correlations between fluxes measured from these eddy covariance towers and GPP estimated from satellite retrievals of SIF, estimates can be made for inputs at the pixels corresponding to each site location.

# Note: this only runs if eval flag switched to TRUE
## Optimize model pars
# Cost function (evaluates error as model vs. obsv, per FME req)
# note that we have to set "pass" to TRUE so SoilR model doesn't fail (neg. resp)
mod.fits.fx <- function(mod, pars, In, sub, lag = 0, upper, lower, cost) {
  
  # start loop
  lapply(seq_along(pars[sub]), function(i) {
    
    # start timer and print PMeco_depth
    start <- Sys.time()
    cat(paste0(names(pars)[sub][i], " parameter fitting\n"))
  
    # define pars
    pars <- pars[sub][[i]]
    if (mod == "2pp") {
      names(pars) <- c("k1", "k2", "gam")
    } else {
      names(pars) <- c("k1", "k2", "tc")
    }
    
    # Set input
    In <- In[sub][[i]]
    
    # define cost function
    if (cost == "14C + cStock") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        cost1 <- modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE)
        cost2 <- modCost(model = modelOutput, obs = obs.resp.14c[sub][[i]], scaleVar = TRUE, cost = cost1) 
        return(modCost(model = modelOutput, obs = obs.cStock[sub][[i]], cost = cost2))
      }
    } else if (cost == "14C + stock/flx") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        cost1 <- modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE)
        cost2 <- modCost(model = modelOutput, obs = obs.resp.14c[sub][[i]], scaleVar = TRUE, cost = cost1) 
        return(modCost(model = modelOutput, obs = obs.flx.stock[[i]], cost = cost2))
      }
    } else if (cost == "14C") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        cost1 <- modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE)
        return(modCost(model = modelOutput, obs = obs.resp.14c[sub][[i]], scaleVar = TRUE, cost = cost1))
      } 
    } else if (cost == "14C bulk + cStock") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        cost1 <- modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE)
        return(modCost(model = modelOutput, obs = obs.cStock[sub][[i]], cost = cost1))
      }
    } else if (cost == "14C bulk only") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        return(modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE))
      }
    }
    
    # fit pars
    fit <- tryCatch(
      modFit(f = mod.Cost,
             p = pars,
             method = 'Nelder-Mead',
             upper = upper, 
             lower = lower),
      error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
    
    Sfun <- sensFun(mod.Cost, fit$par)
    
    # End timer and print elapsed time
    end <- Sys.time()
    cat(paste0("time: ", end - start, "\n"))
    
    # Return fitted parameters and sensitivity
    return(list(modfit = fit, sens = Sfun))
  }) 
}

## 2pp
# par range [0, 1] for all pars
mod.sens.fits.2pp <- mod.fits.fx(mod = "2pp",
                                 pars = pars.i.2pp,
                                 In = in.i,
                                 sub = ix.10,
                                 upper = c(1, 1, 1),
                                 lower = c(0, 0, 0),
                                 cost = "14C")
names(mod.sens.fits.2pp) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp <- lapply(mod.sens.fits.2pp, function(x) x[[1]])
# constrain gamma to [0.5, 0.95]
mod.sens.fits.2pp.p3.5.95 <- mod.fits.fx(mod = "2pp",
                                    pars = pars.i.2pp,
                                    sub = ix.10,
                                    In = in.i,
                                    upper = c(1, 1, 0.951),
                                    lower = c(0, 0, 0.5))
names(mod.sens.fits.2pp.p3.5.95) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp.p3.5.95, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp.p3.5.95", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp.p3.5.95 <- lapply(mod.sens.fits.2pp.p3.5.95, function(x) x[[1]])

# 2pp3 (par range constraints, inputs fit to meas stocks)
mod.sens.fits.2pp3 <- mod.fits.fx(mod = "2pp",
                                  pars = pars.i.2pp,
                                  sub = ix.10,
                                  In = in.meas.2pp,
                                  upper = c(1, .02, .951),
                                  lower = c(.04, .0001, .5),
                                  cost = "14C only")
names(mod.sens.fits.2pp3) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp3, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp3", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp3 <- lapply(mod.sens.fits.2pp3, function(x) x[[1]])
# 2pp3s (par range constraints, inputs fit to meas stocks, + stock constraint)
mod.sens.fits.2pp3s <- mod.fits.fx(mod = "2pp",
                                   pars = pars.i.2pp,
                                   sub = ix.10,
                                   In = in.meas.2pp,
                                   upper = c(1, .02, .951),
                                   lower = c(.04, .0001, .5),
                                   cost = "cStock")
names(mod.sens.fits.2pp3s) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp3s, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp3s", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp3s <- lapply(mod.sens.fits.2pp3s, function(x) x[[1]])

## 2ps
# par range [0, 1] for all pars
mod.sens.fits.2ps <- mod.fits.fx(mod = "2ps",
                                 pars = pars.i.2ps, 
                                 sub = ix.10,
                                 In = in.i,
                                 upper = c(1, 1, 1),
                                 lower = c(0, 0, 0),
                                 cost = "14C")
names(mod.sens.fits.2ps) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps <- lapply(mod.sens.fits.2ps, function(x) x[[1]])

# par range [0, 1] for all pars, stocks + 14C, w/ estimated inputs
# 10
mod.sens.fits.2ps5.10 <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps, 
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, 1, 1),
                                     lower = c(0, 0, 0),
                                     cost = "14C + cStock")
names(mod.sens.fits.2ps5.10) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps5.10, file = paste0("../data/derived/modFit_pars/", "mod.sens.fits.2ps5.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps5.10 <- lapply(mod.sens.fits.2ps5.10, function(x) x[[1]])
# 20
mod.sens.fits.2ps5.20 <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps, 
                                     sub = ix.20,
                                     In = in.est,
                                     upper = c(1, 1, 1),
                                     lower = c(0, 0, 0),
                                     cost = "14C + cStock")
names(mod.sens.fits.2ps5.20) <- names(pars.i.2ps)[ix.20]
save(mod.sens.fits.2ps5.20, file = paste0("../data/derived/modFit_pars/", "mod.sens.fits.2ps5.20", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps5.20 <- lapply(mod.sens.fits.2ps5.20, function(x) x[[1]])

# 20-30
mod.sens.fits.2ps.30 <- mod.fits.fx(mod = "2ps",
                                    pars = pars.i.2ps, 
                                    sub = ix.30,
                                    In = in.i,
                                    upper = c(1, 1, 1),
                                    lower = c(0, 0, 0),
                                    cost = "14C")
names(mod.sens.fits.2ps.30) <- names(pars.i.2ps)[ix.30]
save(mod.sens.fits.2ps.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps.30 <- lapply(mod.sens.fits.2ps.30, function(x) x[[1]])

# 2ps3 (par range constraints, inputs fit to meas stocks)
mod.sens.fits.2ps3 <- mod.fits.fx(mod = "2ps",
                                  pars = pars.i.2ps,
                                  sub = ix.10,
                                  In = in.meas.2ps,
                                  upper = c(1, 1, .15),
                                  lower = c(0, 0, .0004),
                                  cost = "14C only")
names(mod.sens.fits.2ps3) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps3, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps3", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps3 <- lapply(mod.sens.fits.2ps3, function(x) x[[1]])
# 2ps3 (par range constraints, inputs fit to meas stocks, + stock constraint)
mod.sens.fits.2ps3s <- mod.fits.fx(mod = "2ps",
                                   pars = pars.i.2ps,
                                   sub = ix.10,
                                   In = in.meas.2ps,
                                   upper = c(1, .02, .1),
                                   lower = c(.04, .0001, 0),
                                   cost = "cStock")
names(mod.sens.fits.2ps3s) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps3s, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps3s", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps3s <- lapply(mod.sens.fits.2ps3s, function(x) x[[1]])

### 2p4 (par range set, stock and bulk 14C costs, GPP-based inputs by eco)
## 2pp
# 0-10
mod.sens.fits.2pp4.10 <- mod.fits.fx(mod = "2pp",
                                     pars = pars.i.2pp,
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, .02, .951),
                                     lower = c(.04, .0001, .5),
                                     cost = "14C bulk + cStock")
names(mod.sens.fits.2pp4.10) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp4.10, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp4.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp4.10 <- lapply(mod.sens.fits.2pp4.10, function(x) x[[1]])
# 20-30
mod.sens.fits.2pp4.30 <- mod.fits.fx(mod = "2pp",
                                     pars = pars.i.2pp,
                                     sub = ix.30,
                                     In = in.est,
                                     upper = c(1, .02, .951),
                                     lower = c(.04, .0001, .5),
                                     cost = "14C bulk + cStock")
names(mod.sens.fits.2pp4.30) <- names(pars.i.2pp)[ix.30]
save(mod.sens.fits.2pp4.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp4.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp4.30 <- lapply(mod.sens.fits.2pp4.30, function(x) x[[1]])
## 2ps
# 0-10
mod.sens.fits.2ps4.10 <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps,
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, .02, .1),
                                     lower = c(.04, .0001, 0),
                                     cost = "14C bulk + cStock")
names(mod.sens.fits.2ps4.10) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps4.10, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps4.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps4.10 <- lapply(mod.sens.fits.2ps4.10, function(x) x[[1]])
# 20-30
mod.sens.fits.2ps4.30 <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps,
                                     sub = ix.30,
                                     In = in.est,
                                     upper = c(1, .02, .1),
                                     lower = c(.04, .0001, 0),
                                     cost = "14C bulk + cStock")
names(mod.sens.fits.2ps4.30) <- names(pars.i.2ps)[ix.30]
save(mod.sens.fits.2ps4.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps4.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps4.30 <- lapply(mod.sens.fits.2ps4.30, function(x) x[[1]])

### 2p4r (par range set, stock, bulk, and respiration 14C costs, GPP-based inputs by eco)
## 2pp
# 0-10
mod.sens.fits.2pp4r.10 <- mod.fits.fx(mod = "2pp",
                                      pars = pars.i.2pp,
                                      sub = ix.10,
                                      In = in.est,
                                      upper = c(1, .02, .951),
                                      lower = c(.04, .0001, .5),
                                      cost = "14C + cStock")
names(mod.sens.fits.2pp4r.10) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp4r.10, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp4r.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp4r.10 <- lapply(mod.sens.fits.2pp4r.10, function(x) x[[1]])
# 20-30
mod.sens.fits.2pp4r.30 <- mod.fits.fx(mod = "2pp",
                                      pars = pars.i.2pp,
                                      sub = ix.30,
                                      In = in.est,
                                      upper = c(1, .02, .951),
                                      lower = c(.04, .0001, .5),
                                      cost = "14C + cStock")
names(mod.sens.fits.2pp4r.30) <- names(pars.i.2pp)[ix.30]
save(mod.sens.fits.2pp4r.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp4r.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp4r.30 <- lapply(mod.sens.fits.2pp4r.30, function(x) x[[1]])
## 2ps
# 0-10
mod.sens.fits.2ps4r.10 <- mod.fits.fx(mod = "2ps",
                                      pars = pars.i.2ps,
                                      sub = ix.10,
                                      In = in.est,
                                      upper = c(1, .02, .1),
                                      lower = c(.04, .0001, 0),
                                      cost = "14C + cStock")
names(mod.sens.fits.2ps4r.10) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps4r.10, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps4r.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps4r.10 <- lapply(mod.sens.fits.2ps4r.10, function(x) x[[1]])
# 20-30
mod.sens.fits.2ps4r.30 <- mod.fits.fx(mod = "2ps",
                                      pars = pars.i.2ps,
                                      sub = ix.30,
                                      In = in.est,
                                      upper = c(1, .02, .1),
                                      lower = c(.04, .0001, 0),
                                      cost = "14C + cStock")
names(mod.sens.fits.2ps4r.30) <- names(pars.i.2ps)[ix.30]
save(mod.sens.fits.2ps4r.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps4r.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps4r.30 <- lapply(mod.sens.fits.2ps4r.30, function(x) x[[1]])
# load initial parameters as needed
if (!exists("pars.i.2pp")) {
 load("../data/derived/modFit_pars/pars.i.2pp_2020-11-16.Rdata") 
}
if (!exists("pars.i.2ps")) {
  load("../data/derived/modFit_pars/pars.i.2ps_2020-11-16.Rdata")  
}

# load fits as needed
if (!exists("mod.fits.2pp")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2pp_2020-11-16.RData")
}
if (!exists("mod.fits.2pp.p3.5.95")) {
  load("../data/derived/modFit_pars/mod.fits.2pp.p3.5.95_2020-11-16.Rdata")  
}
if (!exists("mod.fits.2ps")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2ps_2020-11-16.Rdata")
}
if (!exists("mod.fits.2pp2")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2pp.flx.stock_2020-12-02.RData")
}
if (!exists("mod.fits.2ps2")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2ps.flx.stock_2020-12-02.Rdata")
}
if (!exists("mod.fits.2pp3")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2pp3_2020-12-08.RData")
}
if (!exists("mod.fits.2ps3")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2ps3_2020-12-08.Rdata")
}
if (!exists("mod.fits.2pp3s")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2pp3s_2020-12-08.RData")
}
if (!exists("mod.fits.2ps3s")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2ps3s_2020-12-08.Rdata")
}

## Par estimates
# 2pp
pars.fit.2pp <- lapply(mod.fits.2pp, "[[", 1)
names(pars.fit.2pp) <- names(pars.i.2pp)[ix.10]
# 2pp gam = [.5, .95]
pars.fit.2pp.p3.5.95 <- lapply(mod.fits.2pp.p3.5.95, "[[", 1)
names(pars.fit.2pp.p3.5.95) <- names(pars.i.2pp)[ix.10]

# 2ps
pars.fit.2ps <- lapply(mod.fits.2ps, "[[", 1)
names(pars.fit.2ps) <- names(pars.i.2ps)[ix.10]


# 2pp2 (input/stock and 14C constraints)
pars.fit.2pp2 <- lapply(mod.fits.2pp2, "[[", 1)
names(pars.fit.2pp2) <- names(pars.i.2pp)[ix.10]
# 2ps2 (input/stock and 14C constraints)
pars.fit.2ps2 <- lapply(mod.fits.2ps2, "[[", 1)
names(pars.fit.2ps2) <- names(pars.i.2ps)[ix.10]

# 2pp3 (14C constraints, constrained par ranges, stock-fit inputs)
pars.fit.2pp3 <- lapply(mod.fits.2pp3, "[[", 1)
Error in lapply(mod.fits.2pp3, "[[", 1) : 
  object 'mod.fits.2pp3' not found
## look at sensFun output
# without constraints
sens.2pp <- lapply(mod.sens.fits.2pp, function(x) x[[2]])
sens.2ps <- lapply(mod.sens.fits.2ps, function(x) x[[2]])
# without stock constraint
sens.2pp3 <- lapply(mod.sens.fits.2pp3, function(x) x[[2]])
sens.2ps3 <- lapply(mod.sens.fits.2ps3, function(x) x[[2]])
# with stock constraint
sens.2pp3s <- lapply(mod.sens.fits.2pp3s, function(x) x[[2]])
sens.2ps3s <- lapply(mod.sens.fits.2ps3s, function(x) x[[2]])
# with stock constraint, w/o resp
sens.2pp4.10 <- lapply(mod.sens.fits.2pp4.10, function(x) x[[2]])
sens.2pp4.30 <- lapply(mod.sens.fits.2pp4.30, function(x) x[[2]])
sens.2ps4.10 <- lapply(mod.sens.fits.2ps4.10, function(x) x[[2]])
sens.2ps4.30 <- lapply(mod.sens.fits.2ps4.30, function(x) x[[2]])
# with stock constraint + resp
sens.2pp4r.10 <- lapply(mod.sens.fits.2pp4r.10, function(x) x[[2]])
sens.2pp4r.30 <- lapply(mod.sens.fits.2pp4r.30, function(x) x[[2]])
sens.2ps4r.10 <- lapply(mod.sens.fits.2ps4r.10, function(x) x[[2]])
sens.2ps4r.30 <- lapply(mod.sens.fits.2ps4r.30, function(x) x[[2]])


# plot sensitivity
# w/o constraints
lapply(sens.2pp, function(x) plot(x, which = c("bulkC", "resp")))
lapply(sens.2ps, function(x) plot(x, which = c("bulkC", "resp")))
# w/o stock constraint
lapply(sens.2pp3, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps3, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2pp3s, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps3s, function(x) plot(x, which = c("bulkC", "cStock")))
# with stock constraint, w/o resp
lapply(sens.2pp4.10, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2pp4.30, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps4.10, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps4.30, function(x) plot(x, which = c("bulkC", "cStock")))
# with stock constraint + resp
lapply(sens.2pp4r.10, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2pp4r.30, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps4r.10, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps4r.30, function(x) plot(x, which = c("bulkC", "cStock")))


# look at identifiability
inden.df.fx <- function(ls, mod) {
  lapply(ls, function(x) {
    df <- collin(x)
    if (mod == "2pp") {
      df$ParCombo <- factor(c("k1 + k2", "k1 + gam", "k2 + gam", "k1 + k2 + gam"))
    } else {
      df$ParCombo <- factor(c("k1 + k2", "k1 + a21", "k2 + a21", "k1 + k2 + a21"))
    }
    return(df)
  })
}

iden.2pp <- inden.df.fx(sens.2pp, mod = "2pp")
iden.2ps <- inden.df.fx(sens.2ps, mod = "2ps")
iden.2pp3 <- inden.df.fx(sens.2pp3, mod = "2pp")
iden.2ps3 <- inden.df.fx(sens.2ps3, mod = "2ps")
iden.2pp3s <- inden.df.fx(sens.2pp3s, mod = "2pp")
iden.2ps3s <- inden.df.fx(sens.2ps3s, mod = "2ps")
# with stock constraint, w/o resp
iden.2pp4.10 <- inden.df.fx(sens.2pp4.10, mod = "2pp")
iden.2pp4.30 <- inden.df.fx(sens.2pp4.30, mod = "2pp")
iden.2ps4.10 <- inden.df.fx(sens.2ps4.10, mod = "2ps")
iden.2ps4.30 <- inden.df.fx(sens.2ps4.30, mod = "2ps")
# with stock constraint + resp
iden.2pp4r.10 <- inden.df.fx(sens.2pp4r.10, mod = "2pp")
iden.2pp4r.30 <- inden.df.fx(sens.2pp4r.30, mod = "2pp")
iden.2ps4r.10 <- inden.df.fx(sens.2ps4r.10, mod = "2ps")
iden.2ps4r.30 <- inden.df.fx(sens.2ps4r.30, mod = "2ps")

# identifiability plot function
coll.plot.fx <- function(df, mod, PMeco_depth, col.max) {
  ggplot(df, aes(N, log(collinearity), color = ParCombo)) +
    geom_hline(yintercept = log(20)) +
    geom_point(size = 3.5, position = position_dodge(width = .1)) +
    scale_y_continuous(limits = c(0, log(col.max))) +
    scale_x_continuous(limits = c(1.5, 3.5), breaks = c(2, 3)) +
    labs(title = paste(PMeco_depth, mod)) +
    theme_bw() +
    theme(panel.grid = element_blank()) +
    if (mod == "2pp" | mod == "2pp + stock") {
     scale_color_manual(
       name = "Parameter combination",
       values = c("k1 + k2" = "#EF476F",
                  "k1 + gam" = "#FFD166",
                  "k2 + gam" = "#118AB2",
                  "k1 + k2 + gam" = "073B4C")) 
    } else {
      scale_color_manual(
        name = "Parameter combination",
        values = c("k1 + k2" = "#EF476F",
                  "k1 + a21" = "#FFD166",
                  "k2 + a21" = "#118AB2",
                  "k1 + k2 + a21" = "073B4C"))
    }
}
lapply(seq_along(iden.2pp), function(i) {
  coll.plot.fx(iden.2pp[[i]], mod = "2pp", names(iden.2pp)[i], max(iden.2pp[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps), function(i) {
  coll.plot.fx(iden.2ps[[i]], mod = "2ps", names(iden.2ps)[i], max(iden.2ps[[i]]["collinearity"]))
})
lapply(seq_along(iden.2pp3), function(i) {
  coll.plot.fx(iden.2pp3[[i]], mod = "2pp", names(iden.2pp3)[i])
})
lapply(seq_along(iden.2pp3s), function(i) {
  coll.plot.fx(iden.2pp3s[[i]], mod = "2pp + stock", names(iden.2pp3s)[i])
})
lapply(seq_along(iden.2ps3), function(i) {
  coll.plot.fx(iden.2ps3[[i]], mod = "2ps", names(iden.2ps3)[i])
})
lapply(seq_along(iden.2ps3s), function(i) {
  coll.plot.fx(iden.2ps3s[[i]], mod = "2ps + stock", names(iden.2ps3s)[i])
})
# stock constraint, w/o resp
col.max <- max(unlist(list(lapply(iden.2ps4.10, function(df) df[["collinearity"]]),
                           lapply(iden.2ps4.30, function(df) df[["collinearity"]]),
                           lapply(iden.2pp4.10, function(df) df[["collinearity"]]),
                           lapply(iden.2pp4.30, function(df) df[["collinearity"]]))))
lapply(seq_along(iden.2ps4.10), function(i) {
  coll.plot.fx(iden.2ps4.10[[i]], mod = "2ps + stock", names(iden.2ps4.10)[i], col.max)
})
lapply(seq_along(iden.2ps4.30), function(i) {
  coll.plot.fx(iden.2ps4.30[[i]], mod = "2ps + stock", names(iden.2ps4.30)[i], col.max)
})
lapply(seq_along(iden.2pp4.10), function(i) {
  coll.plot.fx(iden.2pp4.10[[i]], mod = "2pp + stock", names(iden.2pp4.10)[i], col.max)
})
lapply(seq_along(iden.2pp4.30), function(i) {
  coll.plot.fx(iden.2pp4.30[[i]], mod = "2pp + stock", names(iden.2pp4.30)[i], col.max)
})

# stock constraint + resp
col.max.r <- max(unlist(list(lapply(iden.2ps4r.10, function(df) df[["collinearity"]]),
                             lapply(iden.2ps4r.30, function(df) df[["collinearity"]]),
                             lapply(iden.2pp4r.10, function(df) df[["collinearity"]]),
                             lapply(iden.2pp4r.30, function(df) df[["collinearity"]]))))
lapply(seq_along(iden.2pp4r.10), function(i) {
  coll.plot.fx(iden.2pp4r.10[[i]], mod = "2pp", names(iden.2pp4r.10)[i], col.max)
})
lapply(seq_along(iden.2pp4r.30), function(i) {
  coll.plot.fx(iden.2pp4r.30[[i]], mod = "2ps", names(iden.2pp4r.30)[i], col.max)
})
lapply(seq_along(iden.2ps4r.10), function(i) {
  coll.plot.fx(iden.2ps4r.10[[i]], mod = "2ps + stock", names(iden.2ps4r.10)[i], col.max)
})
lapply(seq_along(iden.2ps4r.30), function(i) {
  coll.plot.fx(iden.2ps4r.30[[i]], mod = "2ps + stock", names(iden.2ps4r.30)[i], col.max)
})
## plot pars
par.plot.fx <- function(mod, depth, par.df, initial = FALSE) {
  par.df %>%
    { if (initial == TRUE) . else filter(., est == "fit") } %>%
    filter(depth == depth) %>%
    pivot_longer(!(est:depth), names_to = "par", values_to = "value") %>%
    mutate(PM = factor(PM),
           eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
    ggplot(., aes(par, value, color = PM, shape = eco)) +
    # geom_jitter(size = 4) +
    geom_point(size = 4, position = position_dodge(width = .5)) +
    scale_color_manual(name = "parent material",
                      labels = c("AN" = "andesite",
                                 "BS" = "basalt",
                                 "GR" = "granite"),
                      values = c("AN" = "blue", 
                                 "BS" = "red", 
                                 "GR" = "darkgray")) +
    facet_wrap(. ~ par, scales = "free") +
    ggtitle(paste0("modFit pars ", mod, " ", depth)) +
    theme_bw() +
    theme(panel.grid.minor = element_blank())
}
# 0-10
# 2pp
par.plot.fx(mod = "2pp",
            depth = "0-10",
            par.df = pars.fit.2pp.df,
            initial = FALSE)
# 2pp, gam = [.5,.95]
par.plot.fx(mod = "2pp (gam = [0.5, 0.95])",
            depth = "0-10",
            par.df = pars.fit.2pp.p3.5.95.df,
            initial = FALSE)
# 2pp2
par.plot.fx(mod = "2pp2",
            depth = "0-10",
            par.df = pars.fit.2pp2.df,
            initial = FALSE)
# 2ps
par.plot.fx(mod = "2ps",
            depth = "0-10",
            par.df = pars.fit.2ps.df,
            initial = FALSE)
# 2ps2
par.plot.fx(mod = "2ps2",
            depth = "0-10",
            par.df = pars.fit.2ps2.df,
            initial = FALSE)

# w/ and w/o stock constraint
par.plot.fx(mod = "2pp3",
            depth = "0-10",
            par.df = pars.fit.2pp3.df,
            initial = FALSE)
par.plot.fx(mod = "2pp3s",
            depth = "0-10",
            par.df = pars.fit.2pp3s.df,
            initial = FALSE)
par.plot.fx(mod = "2ps3",
            depth = "0-10",
            par.df = pars.fit.2ps3.df,
            initial = FALSE)
par.plot.fx(mod = "2ps3s",
            depth = "0-10",
            par.df = pars.fit.2ps3s.df,
            initial = FALSE)

## flux est inputs by eco
# stock and bulk 14C only
par.plot.fx(mod = "2pp4",
            depth = "0-10",
            par.df = pars.fit.2pp4.10.df,
            initial = FALSE)
par.plot.fx(mod = "2pp4",
            depth = "20-30",
            par.df = pars.fit.2pp4.30.df,
            initial = FALSE)
par.plot.fx(mod = "2ps4",
            depth = "0-10",
            par.df = pars.fit.2ps4.10.df,
            initial = FALSE)
par.plot.fx(mod = "2ps4",
            depth = "20-30",
            par.df = pars.fit.2ps4.30.df,
            initial = FALSE)

# stock and bulk + resp 14C
par.plot.fx(mod = "2pp4r",
            depth = "0-10",
            par.df = pars.fit.2pp4r.10.df,
            initial = FALSE)
par.plot.fx(mod = "2pp4r",
            depth = "20-30",
            par.df = pars.fit.2pp4r.30.df,
            initial = FALSE)
par.plot.fx(mod = "2ps4r",
            depth = "0-10",
            par.df = pars.fit.2ps4r.10.df,
            initial = FALSE)
par.plot.fx(mod = "2ps4r",
            depth = "20-30",
            par.df = pars.fit.2ps4r.30.df,
            initial = FALSE)
## Find best inputs
# 2pp
in.fit.2pp <- lapply(seq_along(pars.fit.2pp), function(i) {
  PMeco_depth <- names(pars.fit.2pp)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.fit.2pp[[i]], in.i[ix.10][[i]], SOC))
})
names(in.fit.2pp) <- names(mod.fits.2pp)
# 2pp gam = [.5, .95]
in.fit.2pp.p3.5.95 <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  PMeco_depth <- names(pars.fit.2pp.p3.5.95)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.fit.2pp.p3.5.95[[i]], in.i[ix.10][[i]], SOC))
})
names(in.fit.2pp.p3.5.95) <- names(mod.fits.2pp.p3.5.95)
# 2pp2
in.fit.2pp2 <- lapply(seq_along(pars.fit.2pp2), function(i) {
  PMeco_depth <- names(pars.fit.2pp2)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.fit.2pp2[[i]], in.flx.stock[[i]], SOC))
})
names(in.fit.2pp2) <- names(mod.fits.2pp2)
# 2ps
in.fit.2ps <- lapply(seq_along(pars.fit.2ps), function(i) {
  PMeco_depth <- names(pars.fit.2ps)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2ps", pars.fit.2ps[[i]], in.i[ix.10][[i]], SOC))
})
names(in.fit.2ps) <- names(mod.fits.2ps)
# 2ps2
in.fit.2ps2 <- lapply(seq_along(pars.fit.2ps2), function(i) {
  PMeco_depth <- names(pars.fit.2ps2)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2ps", pars.fit.2ps2[[i]], in.flx.stock[[i]], SOC))
})
names(in.fit.2ps2) <- names(mod.fits.2ps2)

## Calc modeled stocks and compare with measured stocks
# 2pp
mod.socs.2pp.ls <- lapply(seq_along(pars.fit.2pp), function(i) {
  soc.fx("2pp", pars.fit.2pp[[i]], in.fit.2pp[[i]])
})
names(mod.socs.2pp.ls) <- names(pars.fit.2pp)
socs.2pp.ls <- mapply(cbind,
                      csoc.19.0_30[ix.10], 
                      lapply(mod.socs.2pp.ls, colSums), 
                      SIMPLIFY = FALSE)
# 2pp gam = [.5, .95]
mod.socs.2pp.p3.5.95.ls <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  soc.fx("2pp", pars.fit.2pp.p3.5.95[[i]], in.fit.2pp.p3.5.95[[i]])
})
names(mod.socs.2pp.p3.5.95.ls) <- names(pars.fit.2pp.p3.5.95)
socs.2pp.p3.5.95ls <- mapply(cbind,
                             csoc.19.0_30[ix.10], 
                             lapply(mod.socs.2pp.p3.5.95.ls, colSums), 
                             SIMPLIFY = FALSE)
# 2pp2
mod.socs.2pp2.ls <- lapply(seq_along(pars.fit.2pp2), function(i) {
  soc.fx("2pp", pars.fit.2pp2[[i]], in.fit.2pp2[[i]])
})
names(mod.socs.2pp2.ls) <- names(pars.fit.2pp2)

# 2ps
mod.socs.2ps.ls <- lapply(seq_along(pars.fit.2ps), function(i) {
  soc.fx("2ps", pars.fit.2ps[[i]], in.fit.2ps[[i]])
})
names(mod.socs.2ps.ls) <- names(pars.fit.2ps)
socs.2ps.ls <- mapply(cbind,
                      csoc.19.0_30[ix.10], 
                      lapply(mod.socs.2ps.ls, colSums), 
                      SIMPLIFY = FALSE)
# 2ps2
mod.socs.2ps2.ls <- lapply(seq_along(pars.fit.2ps2), function(i) {
  soc.fx("2ps", pars.fit.2ps2[[i]], in.fit.2ps2[[i]])
})
names(mod.socs.2ps2.ls) <- names(pars.fit.2ps2)

## stock and bulk 14C costs only
# 2pp
mod.socs.2pp4.10.ls <- lapply(seq_along(pars.fit.2pp4.10), function(i) {
  soc.fx("2pp", pars.fit.2pp4.10[[i]], in.est[ix.10][[i]])
})
names(mod.socs.2pp4.10.ls) <- names(pars.fit.2pp4.10)
mod.socs.2pp4.30.ls <- lapply(seq_along(pars.fit.2pp4.30), function(i) {
  soc.fx("2pp", pars.fit.2pp4.30[[i]], in.est[ix.30][[i]])
})
names(mod.socs.2pp4.30.ls) <- names(pars.fit.2pp4.30)
# 2ps
mod.socs.2ps4.10.ls <- lapply(seq_along(pars.fit.2ps4.10), function(i) {
  soc.fx("2ps", pars.fit.2ps4.10[[i]], in.est[ix.10][[i]])
})
names(mod.socs.2ps4.10.ls) <- names(pars.fit.2ps4.10)
mod.socs.2ps4.30.ls <- lapply(seq_along(pars.fit.2ps4.30), function(i) {
  soc.fx("2ps", pars.fit.2ps4.30[[i]], in.est[ix.30][[i]])
})
names(mod.socs.2ps4.30.ls) <- names(pars.fit.2ps4.30)

## stock and bulk + resp 14C costs
# 2pp
mod.socs.2pp4r.10.ls <- lapply(seq_along(pars.fit.2pp4r.10), function(i) {
  soc.fx("2pp", pars.fit.2pp4r.10[[i]], in.est[ix.10][[i]])
})
names(mod.socs.2pp4r.10.ls) <- names(pars.fit.2pp4r.10)
mod.socs.2pp4r.30.ls <- lapply(seq_along(pars.fit.2pp4r.30), function(i) {
  soc.fx("2pp", pars.fit.2pp4r.30[[i]], in.est[ix.30][[i]])
})
names(mod.socs.2pp4r.30.ls) <- names(pars.fit.2pp4r.30)
# 2ps
mod.socs.2ps4r.10.ls <- lapply(seq_along(pars.fit.2ps4r.10), function(i) {
  soc.fx("2ps", pars.fit.2ps4r.10[[i]], in.est[ix.10][[i]])
})
names(mod.socs.2ps4r.10.ls) <- names(pars.fit.2ps4r.10)
mod.socs.2ps4r.30.ls <- lapply(seq_along(pars.fit.2ps4r.30), function(i) {
  soc.fx("2ps", pars.fit.2ps4r.30[[i]], in.est[ix.30][[i]])
})
names(mod.socs.2ps4r.30.ls) <- names(pars.fit.2ps4r.30)


## Return data frames of model fits with adjusted inputs and optimal parameters
# 2pp
Twopp.fits <- lapply(seq_along(pars.fit.2pp), function(i) {
  par.fx(pars.fit.2pp[[i]], in.fit.2pp[[i]], verbose = FALSE, mod = "2pp")
})
names(Twopp.fits) <- names(pars.fit.2pp)
# 2pp gam = [.5, .95]
Twopp.p3.5.95.fits <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  par.fx(pars.fit.2pp.p3.5.95[[i]], in.fit.2pp.p3.5.95[[i]], verbose = FALSE, mod = "2pp")
})
names(Twopp.p3.5.95.fits) <- names(pars.fit.2pp.p3.5.95)
# 2pp2
Twopp2.fits <- lapply(seq_along(pars.fit.2pp2), function(i) {
  par.fx(pars.fit.2pp2[[i]], in.fit.2pp2[[i]], verbose = FALSE, mod = "2pp")
})
names(Twopp2.fits) <- names(pars.fit.2pp2)
# 2ps
Twops.fits <- lapply(seq_along(pars.fit.2ps), function(i) {
  par.fx(pars.fit.2ps[[i]], in.fit.2ps[[i]], verbose = FALSE, mod = "2ps")
})
names(Twops.fits) <- names(pars.fit.2ps)
# 2ps2
Twops2.fits <- lapply(seq_along(pars.fit.2ps2), function(i) {
  par.fx(pars.fit.2ps2[[i]], in.fit.2ps2[[i]], verbose = FALSE, mod = "2ps", pass = TRUE)
})
names(Twops2.fits) <- names(pars.fit.2ps2)

## stock and bulk 14C costs only
# 2pp
Twopp4.10.fits <- lapply(seq_along(pars.fit.2pp4.10), function(i) {
  par.fx(pars.fit.2pp4.10[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2pp", pass = FALSE)
})
names(Twopp4.10.fits) <- names(pars.fit.2pp4.10)
Twopp4.30.fits <- lapply(seq_along(pars.fit.2pp4.30), function(i) {
  par.fx(pars.fit.2pp4.30[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2pp", pass = FALSE)
})
names(Twopp4.30.fits) <- names(pars.fit.2pp4.30)
# 2ps
Twops4.10.fits <- lapply(seq_along(pars.fit.2ps4.10), function(i) {
  par.fx(pars.fit.2ps4.10[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE)
})
names(Twops4.10.fits) <- names(pars.fit.2ps4.10)
Twops4.30.fits <- lapply(seq_along(pars.fit.2ps4.30), function(i) {
  par.fx(pars.fit.2ps4.30[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2ps", pass = FALSE)
})
names(Twops4.30.fits) <- names(pars.fit.2ps4.30)

## stock and bulk + resp 14C costs
# 2pp
Twopp4r.10.fits <- lapply(seq_along(pars.fit.2pp4r.10), function(i) {
  par.fx(pars.fit.2pp4r.10[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2pp", pass = FALSE)
})
names(Twopp4r.10.fits) <- names(pars.fit.2pp4r.10)
Twopp4r.30.fits <- lapply(seq_along(pars.fit.2pp4r.30), function(i) {
  par.fx(pars.fit.2pp4r.30[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2pp", pass = FALSE)
})
names(Twopp4r.30.fits) <- names(pars.fit.2pp4r.30)
# 2ps
Twops4r.10.fits <- lapply(seq_along(pars.fit.2ps4r.10), function(i) {
  par.fx(pars.fit.2ps4r.10[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE)
})
names(Twops4r.10.fits) <- names(pars.fit.2ps4r.10)
Twops4r.30.fits <- lapply(seq_along(pars.fit.2ps4r.30), function(i) {
  par.fx(pars.fit.2ps4r.30[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2ps", pass = FALSE)
})
names(Twops4r.30.fits) <- names(pars.fit.2ps4r.30)
# Plot optimized model SOC stocks
mod.socs.df.fx <- function(mod, mod.socs.ls, pools) {
  n <- vapply(mod.socs.ls, nrow, numeric(1))
  return(data.frame(SOC = do.call(rbind, mod.socs.ls),
                    pool = rep(pools, length(mod.socs.ls)),
                    PMeco_depth = rep(names(mod.socs.ls), n),
                    Model = rep(mod, sum(n))))       
}
# run fx
# mod.socs.2p.df <- rbind(mod.socs.df.fx("2pp", mod.socs.2pp.ls, c("fast", "slow"))
#                         ,mod.socs.df.fx("2ps", mod.socs.2ps.ls, c("fast", "slow"))
#                         ,mod.socs.df.fx("2pp [.5,.95]", mod.socs.2pp.p3.5.95.ls, c("fast", "slow"))
#                         ,mod.socs.df.fx("2ps2", mod.socs.2ps2.ls, c("fast", "slow"))
#                         ,mod.socs.df.fx("2pp2", mod.socs.2pp2.ls, c("fast", "slow"))
#                         )
mod.socs.2p.df <- rbind(mod.socs.df.fx("2pp", mod.socs.2pp.ls, c("fast", "slow"))
                        ,mod.socs.df.fx("2ps", mod.socs.2ps.ls, c("fast", "slow"))
                        )


# stocks and bulk 14C only
mod.socs.2p4.10.df <- rbind(mod.socs.df.fx("2pp4 0-10", mod.socs.2pp4.10.ls, c("fast", "slow")), mod.socs.df.fx("2ps4 0-10", mod.socs.2ps4.10.ls, c("fast", "slow")))
mod.socs.2p4.30.df <- rbind(mod.socs.df.fx("2pp4 20-30", mod.socs.2pp4.30.ls, c("fast", "slow")) ,mod.socs.df.fx("2ps4 20-30", mod.socs.2ps4.30.ls, c("fast", "slow")))

# stocks and bulk + resp 14C
mod.socs.2p4r.10.df <- rbind(mod.socs.df.fx("2pp4r 0-10", mod.socs.2pp4r.10.ls, c("fast", "slow")), mod.socs.df.fx("2ps4r 0-10", mod.socs.2ps4r.10.ls, c("fast", "slow")))
mod.socs.2p4r.30.df <- rbind(mod.socs.df.fx("2pp4r 20-30", mod.socs.2pp4r.30.ls, c("fast", "slow")) ,mod.socs.df.fx("2ps4r 20-30", mod.socs.2ps4r.30.ls, c("fast", "slow")))

# combine inputs to compare
# in.fits.df <- pivot_longer(do.call(bind_rows, list(in.fit.2pp,
#                                                 in.fit.2pp.p3.5.95,
#                                                 in.fit.2pp2,
#                                                 in.fit.2ps,
#                                                 in.fit.2ps2)),
#                            everything(),
#                            names_to = "PMeco_depth",
#                            values_to = "inputs")
# in.fits.df$mod <- rep(c("2pp",
#                         "2pp.5.95",
#                         "2pp2",
#                         "2ps",
#                         "2ps2"),
#                       each = 9)
in.fits.df <- pivot_longer(do.call(bind_rows, list(in.fit.2pp,
                                                   in.fit.2ps)),
                           everything(),
                           names_to = "PMeco_depth",
                           values_to = "inputs")
in.fits.df$mod <- rep(c("2pp",
                        "2ps"),
                      each = 9)
                        
## plot stocks
# stock and bulk 14C only
mod.socs.2p4.10.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
mod.socs.2p4.30.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# stock and bulk + resp 14C
mod.socs.2p4r.10.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
mod.socs.2p4r.30.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# inputs
in.fits.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         # Model = factor(Model, levels = c("2pp [.5,.95]", "2pp", "2ps")),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(mod, inputs, fill = mod)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# plot fx
Twop.fit.plot.fx <- function(fit1, fit1.name, fit2, fit2.name, fit3 = NULL, fit3.name = NULL) {
  lapply(seq_along(fit1), function(i) {
    PMeco <- substr(names(fit1)[i], 1, 4)
    lyr_bot <- substr(names(fit1)[i], 
                      nchar(names(fit1)[i]) - 1, 
                      nchar(names(fit1)[i]))
    lyr_top <- ifelse(lyr_bot == 10, 0, ifelse(lyr_bot == 20, 10, 20))
    PMeco_depth <- names(fit1)[i]
    con.df <- con.df.fx(PMeco_depth)
    plot.df <- rbind(fit1[[i]],
                     fit2[[i]],
                     fit3[[i]])
    plot.df$Model <- factor(c(rep(fit1.name, nrow(fit1[[i]])),
                              rep(fit2.name, nrow(fit2[[i]])),
                              rep(fit3.name, nrow(fit3[[i]]))),
                            levels = c(fit1.name, fit2.name, fit3.name))
    return(plot.df %>%
             filter(pool == "bulk C" | pool == "respiration" | pool == "atm") %>%
             ggplot(., aes(years, d14C, color = pool)) +
             geom_path(aes(linetype = Model)) +
             geom_point(data = con.df, aes(Year, d14c, color = pool), size = 3) +
             scale_color_manual(
               name = "Model pool",
               values = c("atm" = 8,
                          "bulk C" = "black",
                          "fast" = "#D81B60",
                          "slow" = "#1E88E5",
                          "respiration" = "#FFC107")) +
             scale_x_continuous(limits = c(1950, 2022)) +
             ggtitle(paste0(PMeco_depth, " 2p mod fits")) +
             xlab("Year") +
             ylab(expression(''*Delta*''^14*'C (‰)')) +
             theme_bw() +
             theme(panel.grid = element_blank()))
  })
}
# 2p modFit optimal model comparison
Twop.fits.plots <- Twop.fit.plot.fx(Twopp.fits, "2pp", Twops.fits, "2ps")
Twop.fits.plots
# Twop.fits.plots2 <- Twop.fit.plot.fx(Twopp.fits, "2pp", Twopp.p3.5.95.fits, "2pp gam = [.5, .95]")
# Twop.fits.plots2
Twop.fits.plots3 <- Twop.fit.plot.fx(Twopp.p3.5.95.fits, "2pp gam = [.5, .95]", Twopp2.fits, "2pp2")
Twop.fits.plots3

## compare fits w/ and w/o resp constraint (2p4 mods)
# 2pp
Twopp4.fits.plots.10 <- Twop.fit.plot.fx(Twopp4.10.fits, "2pp4 0-10cm w/o resp", Twopp4r.10.fits, "2pp4r 0-10cm w/ resp")
Twopp4.fits.plots.30 <- Twop.fit.plot.fx(Twopp4.30.fits, "2pp4 20-30cm w/o resp", Twopp4r.10.fits, "2pp4r 20-30cm w/ resp")
# 2ps
Twops4.fits.plots.10 <- Twop.fit.plot.fx(Twops4.10.fits, "2ps4 0-10cm w/o resp", Twops4r.10.fits, "2ps4r 0-10cm w/ resp")
Twops4.fits.plots.30 <- Twop.fit.plot.fx(Twops4.30.fits, "2ps4 20-30cm w/o resp", Twops4r.10.fits, "2ps4r 20-30cm w/ resp")
# plot
Twopp4.fits.plots.10
Twopp4.fits.plots.30
Twops4.fits.plots.10
Twops4.fits.plots.30
p <- sra.ts.all %>%
    filter(d14c > -200) %>%
    filter(ECO != "rf") %>%
    filter(lyr_bot == 20) %>%
    filter(year != 2009) %>%
    ggplot(., aes(year, d14c)) +
    geom_path(data = atm.14c) +
    geom_point(aes(color = pm, shape = ecoType), size = 3.5) +
    geom_path(aes(color = pm, linetype = Type), size = 1, alpha = 0.3) +
    geom_errorbar(
        aes(ymin = d14c_l, 
            ymax = d14c_u,
            color = pm), 
        width = .5) +
    scale_color_manual(name = "Parent material",
                       values = c("andesite" = "blue", 
                                  "basalt" = "red", 
                                  "granite" = "darkgray")) +
    scale_shape_manual(name = "Ecosystem (type)",
                       values = c("warm (inc)" = 0,
                                  "cool (inc)" = 1,
                                  "cold (inc)" = 2,
                                  "warm (bulk)" = 15,
                                  "cool (bulk)" = 16,
                                  "cold (bulk)" = 17)) +
    facet_grid(rows = vars(eco), cols = vars(pm)) +
    ylab(expression(Delta*''^14*'C (‰)')) +
    xlab("Year") +
    ggtitle("Bulk/inc 10-20 cm") +
    theme_bw() +
    theme(panel.grid = element_blank(),
          axis.text.x = element_text(size = 8))
ggsave("sra.ts.ppwf20.blk.inc.pdf", p, dpi = 300, width = 6.97, height = 5, units = "in")
# inc/bulk profiles
p <- sra.19.01.09 %>%
  filter(lyr_bot < 31) %>%
  select(Year, PM, ECO, PMeco, lyr_bot, d14c, d14c_sd) %>%
  mutate(Type = "bulk",
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         year = as.numeric(as.character(Year))) %>%
  select(-d14c_sd) %>%
  bind_rows(.,
            sra.19.01.inc %>%
              select(year, PM, ECO, PMeco, lyr_bot, d14c, d14c_min, d14c_max) %>%
              rename(d14c_l = d14c_min,
                     d14c_u = d14c_max) %>%
              mutate(Type = "inc")
  ) %>%
  mutate(depth = factor(lyr_bot),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite")),
         ecoType = paste0(eco, " (", Type, ")"))
ggsave("sra.ts.ppwf20.blk.pdf", p, dpi = 300, width = 6.97, height = 5, units = "in")
### Run modfit
## 14C bulk only
# 0-10
mod.sens.fits.2ps.10b <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps, 
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, .02, .999),
                                     lower = c(.02, .0001, .001),
                                     cost = "14C bulk only")
names(mod.sens.fits.2ps.10b) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps.10b, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.10b", "_", Sys.Date(), ".Rdata"))
# # 20-30
# mod.sens.fits.2ps.30b <- mod.fits.fx(mod = "2ps",
#                                      pars = pars.i.2ps, 
#                                      sub = ix.30,
#                                      In = in.est,
#                                      upper = c(1, .02, .15),
#                                      lower = c(.005, .0001, .0004),
#                                      cost = "14C bulk only")
# names(mod.sens.fits.2ps.30b) <- names(pars.i.2ps)[ix.30]
# save(mod.sens.fits.2ps.30b, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30b", "_", Sys.Date(), ".Rdata"))
# mod.fits.2ps.30b <- lapply(mod.sens.fits.2ps.30b, function(x) x[[1]])

## 14C (bulk + resp)
# 0-10
mod.sens.fits.2ps.10br <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps, 
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, .02, .999),
                                     lower = c(.02, .0001, .001),
                                     cost = "14C")
names(mod.sens.fits.2ps.10br) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps.10br, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.10br", "_", Sys.Date(), ".Rdata"))
# 10-20, lag = 5
mod.sens.fits.2ps.20br.l <- mod.fits.fx(mod = "2ps",
                                       pars = pars.i.2ps,
                                       sub = ix.20,
                                       In = in.est,
                                      lag = 5,
                                       upper = c(1, .02, .99),
                                       lower = c(.02, .0001, .01),
                                       cost = "14C")
names(mod.sens.fits.2ps.20br.l) <- names(pars.i.2ps)[ix.20]
save(mod.sens.fits.2ps.20br.l, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.20br.l", "_", Sys.Date(), ".Rdata"))
# # 20-30
# mod.sens.fits.2ps.30br <- mod.fits.fx(mod = "2ps",
#                                      pars = pars.i.2ps, 
#                                      sub = ix.30,
#                                      In = in.est,
#                                      upper = c(1, .02, .15),
#                                      lower = c(.005, .0001, .0004),
#                                      cost = "14C")
# names(mod.sens.fits.2ps.30br) <- names(pars.i.2ps)[ix.30]
# save(mod.sens.fits.2ps.30br, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30br", "_", Sys.Date(), ".Rdata"))
# mod.fits.2ps.30br <- lapply(mod.sens.fits.2ps.30br, function(x) x[[1]])

## 14C bulk + stocks
# 0-10
mod.sens.fits.2ps.10bs <- mod.fits.fx(mod = "2ps",
                                      pars = pars.i.2ps,
                                      sub = ix.10,
                                      In = in.est,
                                      upper = c(1, .02, .999),
                                      lower = c(.02, .0001, .001),
                                      cost = "14C bulk + cStock")
names(mod.sens.fits.2ps.10bs) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps.10bs, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.10bs", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps.10bs <- lapply(mod.sens.fits.2ps.10bs, function(x) x[[1]])
# # 20-30
# mod.sens.fits.2ps.30b <- mod.fits.fx(mod = "2ps",
#                                      pars = pars.i.2ps, 
#                                      sub = ix.30,
#                                      In = in.est,
#                                      upper = c(1, .02, .15),
#                                      lower = c(.005, .0001, .0004),
#                                      cost = "14C bulk only")
# names(mod.sens.fits.2ps.30b) <- names(pars.i.2ps)[ix.30]
# save(mod.sens.fits.2ps.30b, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30b", "_", Sys.Date(), ".Rdata"))
# mod.fits.2ps.30b <- lapply(mod.sens.fits.2ps.30b, function(x) x[[1]])

## 14C + cStock (14C resp, 14C bulk, stocks)
# 0-10
mod.sens.fits.2ps.10rbs <- mod.fits.fx(mod = "2ps",
                                       pars = pars.i.2ps,
                                       sub = ix.10,
                                       In = in.est,
                                       upper = c(1, .02, .999),
                                       lower = c(.02, .0001, .001),
                                       cost = "14C + cStock")
names(mod.sens.fits.2ps.10rbs) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps.10rbs, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.10rbs", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps.10rbs <- lapply(mod.sens.fits.2ps.10rbs, function(x) x[[1]])
# 10-20
# w/o lag
mod.sens.fits.2ps.20rbs <- mod.fits.fx(mod = "2ps",
                                       pars = pars.i.2ps,
                                       sub = ix.20,
                                       In = in.est,
                                       upper = c(1, .02, .99),
                                       lower = c(.02, .0001, .01),
                                       cost = "14C + cStock")
names(mod.sens.fits.2ps.20rbs) <- names(pars.i.2ps)[ix.20]
save(mod.sens.fits.2ps.20rbs, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.20rbs", "_", Sys.Date(), ".Rdata"))
# w/ lag = 12
mod.sens.fits.2ps.20rbs.l <- mod.fits.fx(mod = "2ps",
                                       pars = pars.i.2ps,
                                       sub = ix.20,
                                       In = in.est,
                                       lag = 12,
                                       upper = c(1, .02, .99),
                                       lower = c(.02, .0001, .01),
                                       cost = "14C + cStock")
names(mod.sens.fits.2ps.20rbs.l) <- names(pars.i.2ps)[ix.20]
save(mod.sens.fits.2ps.20rbs.l, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.20rbs.l", "_", Sys.Date(), ".Rdata"))
# # 20-30
# mod.sens.fits.2ps.30b <- mod.fits.fx(mod = "2ps",
#                                      pars = pars.i.2ps, 
#                                      sub = ix.30,
#                                      In = in.est,
#                                      upper = c(1, .02, .15),
#                                      lower = c(.005, .0001, .0004),
#                                      cost = "14C bulk only")
# names(mod.sens.fits.2ps.30b) <- names(pars.i.2ps)[ix.30]
# save(mod.sens.fits.2ps.30b, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30b", "_", Sys.Date(), ".Rdata"))
# mod.fits.2ps.30b <- lapply(mod.sens.fits.2ps.30b, function(x) x[[1]])
# SAB fits
load("../data/derived/modFit_pars/mod.fits.2ps.10b_2021-04-07.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.10br_2021-04-07.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.10bs_2021-04-07.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.10rbs_2021-04-07.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.20rbs_2021-04-12.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.20rbs.l_2021-04-13.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.20br.l_2021-04-13.Rdata")
load("../data/derived/modFit_pars/pars.i.2ps_2021-04-06.Rdata")

# extract mod fits
mod.fits.2ps.10b <- lapply(mod.sens.fits.2ps.10b, function(x) x[[1]])
mod.fits.2ps.10bs <- lapply(mod.sens.fits.2ps.10bs, function(x) x[[1]])
mod.fits.2ps.10br <- lapply(mod.sens.fits.2ps.10br, function(x) x[[1]])
mod.fits.2ps.10rbs <- lapply(mod.sens.fits.2ps.10rbs, function(x) x[[1]])
mod.fits.2ps.20rbs <- lapply(mod.sens.fits.2ps.20rbs, function(x) x[[1]])
mod.fits.2ps.20rbs.l <- lapply(mod.sens.fits.2ps.20rbs.l, function(x) x[[1]])
mod.fits.2ps.20br.l <- lapply(mod.sens.fits.2ps.20br.l, function(x) x[[1]]) 
  
# Sensitivity/Identifiability
#####
# extract at sensFun output
sens.2ps.10b <- lapply(mod.sens.fits.2ps.10b, function(x) x[[2]])
sens.2ps.10br <- lapply(mod.sens.fits.2ps.10br, function(x) x[[2]])
sens.2ps.10bs <- lapply(mod.sens.fits.2ps.10bs, function(x) x[[2]])
sens.2ps.10rbs <- lapply(mod.sens.fits.2ps.10rbs, function(x) x[[2]])
sens.2ps.20rbs <- lapply(mod.sens.fits.2ps.20rbs, function(x) x[[2]])
sens.2ps.20rbs.l <- lapply(mod.sens.fits.2ps.20rbs.l, function(x) x[[2]])
sens.2ps.20br.l <- lapply(mod.sens.fits.2ps.20br.l, function(x) x[[2]])

# plot sensitivity
lapply(sens.2ps.10b, function(x) plot(x, which = c("bulkC", "resp")))
lapply(sens.2ps.10br, function(x) plot(x, which = c("bulkC", "resp")))
lapply(sens.2ps.10bs, function(x) plot(x, which = c("bulkC", "resp")))
lapply(sens.2ps.10rbs, function(x) plot(x, which = c("bulkC", "resp")))

# look at identifiability
iden.2ps.10b <- inden.df.fx(sens.2ps.10b, mod = "2ps")
iden.2ps.10br <- inden.df.fx(sens.2ps.10br, mod = "2ps")
iden.2ps.10bs <- inden.df.fx(sens.2ps.10bs, mod = "2ps")
iden.2ps.10rbs <- inden.df.fx(sens.2ps.10rbs, mod = "2ps")
iden.2ps.20rbs <- inden.df.fx(sens.2ps.20rbs, mod = "2ps")
iden.2ps.20rbs.l <- inden.df.fx(sens.2ps.20rbs.l, mod = "2ps")
iden.2ps.20br.l <- inden.df.fx(sens.2ps.20br.l, mod = "2ps")

# plot
lapply(seq_along(iden.2ps.10bs), function(i) {
  coll.plot.fx(iden.2ps.10bs[[i]], mod = "2ps", 
               names(iden.2ps.10bs)[i], 
               max(iden.2ps.10bs[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.10br), function(i) {
  coll.plot.fx(iden.2ps.10br[[i]], mod = "2ps", 
               names(iden.2ps.10br)[i], 
               max(iden.2ps.10br[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.10rbs), function(i) {
  coll.plot.fx(iden.2ps.10rbs[[i]], mod = "2ps", 
               names(iden.2ps.10rbs)[i], 
               max(iden.2ps.10rbs[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.20rbs), function(i) {
  coll.plot.fx(iden.2ps.20rbs[[i]], mod = "2ps", 
               names(iden.2ps.20rbs)[i], 
               max(iden.2ps.20rbs[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.20rbs.l), function(i) {
  coll.plot.fx(iden.2ps.20rbs.l[[i]], mod = "2ps", 
               names(iden.2ps.20rbs.l)[i], 
               max(iden.2ps.20rbs.l[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.20br.l), function(i) {
  coll.plot.fx(iden.2ps.20br.l[[i]], mod = "2ps", 
               names(iden.2ps.20br.l)[i], 
               max(iden.2ps.20br.l[[i]]["collinearity"]))
})
#####

# Extract optimized pars from modfit output
#####
## bulk 14c only
# 0-10
pars.fit.2ps.10b <- lapply(mod.fits.2ps.10b, "[[", 1)
names(pars.fit.2ps.10b) <- names(pars.i.2ps)[ix.10]
# # 20-30
# pars.fit.2ps.30b <- lapply(mod.fits.2ps.30b, "[[", 1)
# names(pars.fit.2ps.30b) <- names(pars.i.2ps)[ix.30]

## resp + bulk 14c
# 0-10
pars.fit.2ps.10br <- lapply(mod.fits.2ps.10br, "[[", 1)
names(pars.fit.2ps.10br) <- names(pars.i.2ps)[ix.10]
# 10-20 w/ lag = 5y
pars.fit.2ps.20br.l <- lapply(mod.fits.2ps.20br.l, "[[", 1)
names(pars.fit.2ps.20br.l) <- names(pars.i.2ps)[ix.20]
# # 20-30
# pars.fit.2ps.30br <- lapply(mod.fits.2ps.30br, "[[", 1)
# names(pars.fit.2ps.30br) <- names(pars.i.2ps)[ix.30]

## bulk 14c + stocks
# 0-10
pars.fit.2ps.10bs <- lapply(mod.fits.2ps.10bs, "[[", 1)
names(pars.fit.2ps.10bs) <- names(pars.i.2ps)[ix.10]
# # 20-30
# pars.fit.2ps.30br <- lapply(mod.fits.2ps.30br, "[[", 1)
# names(pars.fit.2ps.30br) <- names(pars.i.2ps)[ix.30]

## resp, bulk 14c, stocks
# 0-10
pars.fit.2ps.10rbs <- lapply(mod.fits.2ps.10rbs, "[[", 1)
names(pars.fit.2ps.10rbs) <- names(pars.i.2ps)[ix.10]
# 10-20
pars.fit.2ps.20rbs <- lapply(mod.fits.2ps.20rbs, "[[", 1)
names(pars.fit.2ps.20rbs) <- names(pars.i.2ps)[ix.20]
# 10-20 w/ lag = 12y
pars.fit.2ps.20rbs.l <- lapply(mod.fits.2ps.20rbs.l, "[[", 1)
names(pars.fit.2ps.20rbs.l) <- names(pars.i.2ps)[ix.20]
# # 20-30
# pars.fit.2ps.30br <- lapply(mod.fits.2ps.30br, "[[", 1)
# names(pars.fit.2ps.30br) <- names(pars.i.2ps)[ix.30]
#####

# SOC stocks
#####
# w/o stock constraint
mod.socs.2ps.10b.ls <- lapply(seq_along(pars.fit.2ps.10b), function(i) {
  soc.fx("2ps", pars.fit.2ps.10b[[i]], in.est[[i]])
})
names(mod.socs.2ps.10b.ls) <- names(pars.fit.2ps.10b)
mod.socs.2ps.10br.ls <- lapply(seq_along(pars.fit.2ps.10br), function(i) {
  soc.fx("2ps", pars.fit.2ps.10br[[i]], in.est[[i]])
})
names(mod.socs.2ps.10br.ls) <- names(pars.fit.2ps.10br)
socs.2ps.10br.ls <- mapply(cbind,
                           csoc.19.0_30[ix.10], 
                           lapply(mod.socs.2ps.10br.ls, colSums), 
                           SIMPLIFY = FALSE)
# w/ stock constraint
mod.socs.2ps.10bs.ls <- lapply(seq_along(pars.fit.2ps.10bs), function(i) {
  soc.fx("2ps", pars.fit.2ps.10bs[[i]], in.est[[i]])
})
names(mod.socs.2ps.10bs.ls) <- names(pars.fit.2ps.10bs)
mod.socs.2ps.10rbs.ls <- lapply(seq_along(pars.fit.2ps.10rbs), function(i) {
  soc.fx("2ps", pars.fit.2ps.10rbs[[i]], in.est[[i]])
})
names(mod.socs.2ps.10rbs.ls) <- names(pars.fit.2ps.10rbs)
mod.socs.2ps.20rbs.ls <- lapply(seq_along(pars.fit.2ps.20rbs), function(i) {
  soc.fx("2ps", pars.fit.2ps.20rbs[[i]], in.est[ix.20][[i]])
})
names(mod.socs.2ps.20rbs.ls) <- names(pars.fit.2ps.20rbs)
socs.2ps.10rbs.ls <- mapply(cbind,
                           csoc.19.0_30[ix.10], 
                           lapply(mod.socs.2ps.10rbs.ls, colSums), 
                           SIMPLIFY = FALSE)

## make df for plotting
# resp + bulk, w/ and w/o stocks
mod.socs.2ps.10brrbs.df <- rbind(mod.socs.df.fx("2ps w/o stock", mod.socs.2ps.10br.ls, c("fast", "slow"))
                                 ,mod.socs.df.fx("2ps w/ stock", mod.socs.2ps.10rbs.ls, c("fast", "slow"))
                                 ,data.frame(SOC = unlist(lapply(csoc.19.0_30[ix.10], "[[", 4)),
                                             PMeco_depth = paste0(
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 1)),
                                               "_",
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 2)),
                                               "-",
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 3))),
                                             Model = "measured",
                                             pool = "total")
                                 )
# bulk + stock, vs. resp, bulk, + stock
mod.socs.2ps.10bsrbs.df <- rbind(mod.socs.df.fx("2ps bulk + stock only", mod.socs.2ps.10bs.ls, c("fast", "slow"))
                                 ,mod.socs.df.fx("2ps bulk, resp, + stock", mod.socs.2ps.10rbs.ls, c("fast", "slow"))
                                 ,data.frame(SOC = unlist(lapply(csoc.19.0_30[ix.10], "[[", 4)),
                                             PMeco_depth = paste0(
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 1)),
                                               "_",
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 2)),
                                               "-",
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 3))),
                                             Model = "measured",
                                             pool = "total")
                                 )


## plot
mod.socs.2ps.10brrbs.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
mod.socs.2ps.10bsrbs.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
#####

### Summarize optimized par data for plotting
## bulk 14c only
# 0-10
pars.fit.2ps.10b.df <- par.fit.df.fx(mod = "2ps",
                                     pars.fit = pars.fit.2ps.10b,
                                     pars.i = pars.i.2ps[ix.10])
# # 20-30
# pars.fit.2ps.30b.df <- par.fit.df.fx(mod = "2ps",
#                                       pars.fit = pars.fit.2ps.30b,
#                                       pars.i = pars.i.2ps[ix.30])

## resp + bulk 14c
# 0-10
pars.fit.2ps.10br.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.10br,
                                       pars.i = pars.i.2ps[ix.10])
# 10-20
pars.fit.2ps.20br.l.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.20br.l,
                                       pars.i = pars.i.2ps[ix.10])
# # 20-30
# pars.fit.2ps.30br.df <- par.fit.df.fx(mod = "2ps",
#                                        pars.fit = pars.fit.2ps.30br,
#                                        pars.i = pars.i.2ps[ix.30])

## bulk 14c + stocks
# 0-10
pars.fit.2ps.10bs.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.10bs,
                                       pars.i = pars.i.2ps[ix.10])
# # 20-30
# pars.fit.2ps.30br.df <- par.fit.df.fx(mod = "2ps",
#                                        pars.fit = pars.fit.2ps.30br,
#                                        pars.i = pars.i.2ps[ix.30])

## resp, bulk, stocks
# 0-10
pars.fit.2ps.10rbs.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.10rbs,
                                       pars.i = pars.i.2ps[ix.10])
# 10-20
pars.fit.2ps.20rbs.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.20rbs,
                                       pars.i = pars.i.2ps[ix.20])
# w/ lag
pars.fit.2ps.20rbs.l.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.20rbs.l,
                                       pars.i = pars.i.2ps[ix.20])
# # 20-30
# pars.fit.2ps.30br.df <- par.fit.df.fx(mod = "2ps",
#                                        pars.fit = pars.fit.2ps.30br,
#                                        pars.i = pars.i.2ps[ix.30])

### Par fits
par.plot.fx(mod = "2ps bulk 14c",
            depth = "0-10",
            par.df = pars.fit.2ps.10b.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp + bulk 14c",
            depth = "0-10",
            par.df = pars.fit.2ps.10br.df,
            initial = FALSE)
par.plot.fx(mod = "2ps bulk 14c + stocks",
            depth = "0-10",
            par.df = pars.fit.2ps.10bs.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp, bulk, stocks",
            depth = "0-10",
            par.df = pars.fit.2ps.10rbs.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp, bulk, stocks",
            depth = "10-20",
            par.df = pars.fit.2ps.20rbs.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp, bulk, stocks",
            depth = "10-20",
            par.df = pars.fit.2ps.20rbs.l.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp, bulk",
            depth = "10-20",
            par.df = pars.fit.2ps.20br.l.df,
            initial = FALSE)
# par.plot.fx(mod = "2ps bulk 14c",
#             depth = "20-30",
#             par.df = pars.fit.2ps.30b.df,
#             initial = FALSE)
# par.plot.fx(mod = "2ps resp + bulk 14c",
#             depth = "20-30",
#             par.df = pars.fit.2ps.30br.df,
#             initial = FALSE)

### Fit models with optimized pars
## bulk 14C only
# 0-10
Twops.10b.fits <- lapply(seq_along(pars.fit.2ps.10b), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.10b[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.10b.fits) <- names(pars.fit.2ps.10b)
# # 20-30
# Twops.30b.fits <- lapply(seq_along(pars.fit.2ps.30b), function(i) {
#   tryCatch(
#     par.fx(pars.fit.2ps.30b[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
#     error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
# })
# names(Twops.30b.fits) <- names(pars.fit.2ps.30b)

## resp + bulk 14C
# 0-10
Twops.10br.fits <- lapply(seq_along(pars.fit.2ps.10br), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.10br[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.10br.fits) <- names(pars.fit.2ps.10br)
# 10-20
Twops.20br.l.fits <- lapply(seq_along(pars.fit.2ps.20br.l), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.20br.l[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.20br.l.fits) <- names(pars.fit.2ps.20br.l)
# # 20-30
# Twops.30br.fits <- lapply(seq_along(pars.fit.2ps.30br), function(i) {
#   tryCatch(
#     par.fx(pars.fit.2ps.30br[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
#     error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
# })
# names(Twops.30br.fits) <- names(pars.fit.2ps.30br)

## bulk 14C + stocks
# 0-10
Twops.10bs.fits <- lapply(seq_along(pars.fit.2ps.10bs), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.10bs[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.10bs.fits) <- names(pars.fit.2ps.10bs)

## resp, bulk, stocks
# 0-10
Twops.10rbs.fits <- lapply(seq_along(pars.fit.2ps.10rbs), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.10rbs[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.10rbs.fits) <- names(pars.fit.2ps.10rbs)
# 10-20
Twops.20rbs.fits <- lapply(seq_along(pars.fit.2ps.20rbs), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.20rbs[[i]], in.est[ix.20][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.20rbs.fits) <- names(pars.fit.2ps.20rbs)

# # remove null entries
# Twops.10b.fits <- Filter(Negate(is.null), Twops.10b.fits)
# Twops.30b.fits <- Filter(Negate(is.null), Twops.30b.fits)
# Twops.10br.fits <- Filter(Negate(is.null), Twops.10br.fits)
# Twops.30br.fits <- Filter(Negate(is.null), Twops.30br.fits)

# Look at role of resp constraint in fit
# lapply(seq_along(Twops.10b.fits), function(i) {
#   C14.2p.plot.fx(Twops.10b.fits[[i]],
#                  con.df = con.df.fx(names(Twops.10b.fits)[i]), 
#                  mod = "2ps bulk only",
#                  PMeco_depth = names(Twops.10b.fits)[i])
# })
lapply(seq_along(Twops.10br.fits), function(i) {
  C14.2p.plot.fx(Twops.10br.fits[[i]], 
                 con.df = con.df.fx(names(Twops.10br.fits)[i]), 
                 mod = "2ps bulk + resp",
                 PMeco_depth = names(Twops.10br.fits)[i])
})
# lapply(seq_along(Twops.10bs.fits), function(i) {
#   C14.2p.plot.fx(Twops.10bs.fits[[i]],
#                  con.df = con.df.fx(names(Twops.10bs.fits)[i]), 
#                  mod = "2ps bulk + stock",
#                  PMeco_depth = names(Twops.10bs.fits)[i])
# })
lapply(seq_along(Twops.10rbs.fits), function(i) {
  C14.2p.plot.fx(Twops.10rbs.fits[[i]], 
                 con.df = con.df.fx(names(Twops.10rbs.fits)[i]), 
                 mod = "bulk, resp, stock",
                 PMeco_depth = names(Twops.10rbs.fits)[i])
})
# 10-20
lapply(seq_along(Twops.20rbs.fits), function(i) {
  C14.2p.plot.fx(Twops.20rbs.fits[[i]], 
                 con.df = con.df.fx(names(Twops.20rbs.fits)[i]), 
                 mod = "bulk, resp, stock",
                 PMeco_depth = names(Twops.20rbs.fits)[i])
})
# lapply(seq_along(Twops.30b.fits), function(i) {
#   C14.2p.plot.fx(Twops.30b.fits[[i]],
#                  con.df = con.df.fx(names(Twops.30b.fits)[i]), 
#                  mod = "2ps bulk only",
#                  PMeco_depth = names(Twops.30b.fits)[i])
# })
# lapply(seq_along(Twops.30br.fits), function(i) {
#   C14.2p.plot.fx(Twops.30br.fits[[i]], 
#                  con.df = con.df.fx(names(Twops.30br.fits)[i]), 
#                  mod = "2ps bulk + resp",
#                  PMeco_depth = names(Twops.30br.fits)[i])
# })

## Show role of resp in constraining models
# GRwf 0-10
Twop.fit.plot.fx(Twops.10bs.fits[which(names(Twops.10bs.fits) == "GRwf_0-10")], 
                 "2ps 0-10cm, bulk 14c + stock", 
                 Twops.10rbs.fits[which(names(Twops.10rbs.fits) == "GRwf_0-10")],
                 "2ps 0-10cm, resp & bulk 14c + stock")
# BSrf 0-10
Twop.fit.plot.fx(Twops.10br.fits[which(names(Twops.10br.fits) == "BSrf_0-10")], 
                 "2ps 0-10cm w/ resp", 
                 Twops.10b.fits[which(names(Twops.10b.fits) == "BSrf_0-10")],
                 "2ps 0-10cm w/o resp")
Twop.fit.plot.fx(Twops.10rbs.fits[which(names(Twops.10rbs.fits) == "BSrf_0-10")],
                 "2ps 0-10cm w/ resp, bulk, stocks",
                 Twops.10bs.fits[which(names(Twops.10bs.fits) == "BSrf_0-10")],
                 "2ps 0-10cm w/o resp (bulk + stocks only)")
Twop.fit.plot.fx(Twops.10rbs.fits[which(names(Twops.10rbs.fits) == "BSwf_10-20")],
                 "Basalt/cool 10-20",
                 Twops.10rbs.fits[which(names(Twops.10rbs.fits) == "GRwf_10-20")],
                 "Granite/cool 10-20")

# compare resp + bulk fits w/ and w/o stocks
Twop.fit.plot.fx(Twops.10rbs.fits, 
                 "2ps 0-10cm w/ resp, bulk, stocks", 
                 Twops.10br.fits,
                 "2ps 0-10cm w/ resp + bulk, no stock")
# compare resp + bulk fits w/ and w/o stocks
Twop.fit.plot.fx(Twops.10bs.fits, 
                 "2ps 0-10cm, bulk 14c + stock", 
                 Twops.10rbs.fits,
                 "2ps 0-10cm, resp & bulk 14c + stock")
# compare BSwf and GRwf 10-20
BSGRwf20.con.df <- cbind(rbind(con.df.fx("BSwf_10-20"), con.df.fx("GRwf_10-20")),
                               pm = factor(rep(c("basalt", "granite"), each = c(11))))
BSGRwf20.con.df <- BSGRwf20.con.df[-which(BSGRwf20.con.df$Year == 2009.5), ]
ANGRwf20.con.df <- cbind(rbind(con.df.fx("BSwf_10-20"), con.df.fx("GRwf_10-20")),
                               pm = factor(rep(c("basalt", "granite"), each = c(11))))
BSGRwf20.con.df <- BSGRwf20.con.df[-which(BSGRwf20.con.df$Year == 2009.5), ]
atm.14c2 <- Twops.20rbs.fits$`BSwf_10-20`[Twops.20rbs.fits$`BSwf_10-20`$years >= 1950 & Twops.20rbs.fits$`BSwf_10-20`$pool == "atm", ]
# plot
p <- rbind(Twops.20rbs.fits$`BSwf_10-20`,
      Twops.20rbs.fits$`GRwf_10-20`) %>%
  mutate(pm = rep(c("basalt", "granite"), 
                  each = nrow(Twops.20rbs.fits$`BSwf_10-20`))) %>%
  filter(pool == "bulk C" | pool == "respiration") %>%
  ggplot(., aes(years, d14C)) +
  geom_path(data = atm.14c2) +
  geom_path(aes(linetype = pool, color = pm)) +
  geom_point(data = BSGRwf20.con.df, 
             aes(Year, d14c, color = pm, shape = pool), 
             size = 2.5,
             position = position_dodge(width = 1)) +
  scale_color_manual(
    name = "Parent material",
    values = c("basalt" = "red",
               "granite" = "darkgray")) +
  scale_shape_manual(
    name = "",
    values = c("bulk C" = 16,
               "respiration" = 1)) +
  scale_linetype_manual(
   name = "Pool",
   values = c("bulk C" = 1,
              "respiration" = 2)) +
  scale_x_continuous(limits = c(1950, 2022)) +
  xlab("Year") +
  ylab(expression(''*Delta*''^14*'C (‰)')) +
  theme_bw() +
  theme(panel.grid = element_blank())
ggsave("sra.2ps.BSGRwf20.pdf", p, dpi = 300, width = 6, height = 5, units = "in")
# compare resp + bulk fits w/ and w/o stocks
Twop.fit.plot.fx(Twops.20br.l.fits, 
                 "2ps 0-10cm, bulk 14c + stock", 
                 Twops.20rbs.fits,
                 "2ps 0-10cm, resp & bulk 14c + stock")
#####

# ages and transit times
#####
# 2ps
SA.2ps.20.rbs.ls <- lapply(seq_along(pars.fit.2ps.20rbs), function(i) {
  ks <- pars.fit.2ps.20rbs[[i]][1:2]
  tc <- pars.fit.2ps.20rbs[[i]][3]
  In <- in.est[ix.20][[i]]
  A <- diag(-ks)
  A[2, 1] <- tc * ks[1]
  return(systemAge(A = A, u = c(In, 0)))
})
names(SA.2ps.20.rbs.ls) <- names(pars.fit.2ps.20rbs)
lapply(SA.2ps.20.rbs.ls, "[[", 1)

## Transit time
# 2ps
TT.2ps.20.rbs.ls <- lapply(seq_along(pars.fit.2ps.20rbs), function(i) {
  ks <- pars.fit.2ps.20rbs[[i]][1:2]
  tc <- pars.fit.2ps.20rbs[[i]][3]
  In <- in.est[ix.20][[i]]
  A <- diag(-ks)
  A[2, 1] <- tc * ks[1]
  return(transitTime(A = A, u = c(In, 0)))
})
names(TT.2ps.20.rbs.ls) <- names(pars.fit.2ps.20rbs)
lapply(TT.2ps.20.rbs.ls, "[[", 1)
# 0-10
TT.MA.2ps.10.rbs.ls <- lapply(seq_along(pars.fit.2ps.10rbs), function(i) {
  ks <- pars.fit.2ps.10rbs[[i]][1:2]
  tc <- pars.fit.2ps.10rbs[[i]][3]
  In <- in.est[ix.10][[i]]
  A <- diag(-ks)
  A[2, 1] <- tc * ks[1]
  TT <- transitTime(A = A, u = c(In, 0))
  Age <- systemAge(A = A, u = c(In, 0))
  return(list(TT = TT$meanTransitTime, Age = Age$meanSystemAge))
})
names(TT.MA.2ps.10.rbs.ls) <- names(pars.fit.2ps.10rbs)
lapply(TT.MA.2ps.10.rbs.ls, unlist)
# 
ageD.2ps.10.rbs.ls <- lapply(seq_along(pars.fit.2ps.10rbs), function(i) {
  ks <- pars.fit.2ps.10rbs[[i]][1:2]
  tc <- pars.fit.2ps.10rbs[[i]][3]
  In <- in.est[ix.10][[i]]
  A <- diag(-ks)
  A[2, 1] <- tc * ks[1]
  return(systemAge(A = A, u = c(In, 0)))
})
names(ageD.2ps.10.rbs.ls) <- names(pars.fit.2ps.10rbs)
# compare output of 2pp and 2ps model fits
merge(ssr.2pp.df, ssr.2ps.df, by = "PMeco_depth", suffixes = c("_2pp", "_2ps")) %>%
  mutate(ssr_2pp = round(ssr_2pp, 1),
         ssr_2ps = round(ssr_2ps, 1),
         dif = ssr_2pp - ssr_2ps)
merge(var_ms.2pp.df,
      var_ms.2ps.df,
      by = c("PMeco_depth", "var"),
      suffixes = c("_2pp", "_2ps")) %>%
  mutate(var_ms_2pp = round(var_ms_2pp, 4),
         var_ms_2ps = round(var_ms_2ps, 4),
         dif = var_ms_2pp - var_ms_2ps)

## Plot
# SSR, PM
rbind(ssr.2pp.df, ssr.2ps.df) %>%
  mutate(mod = rep(c("2pp", "2ps"), each = nrow(ssr.2pp.df)),
         PM = substr(PMeco_depth, 1, 2),
         eco = substr(PMeco_depth, 3, 4)) %>%
  group_by(PM, mod) %>%
  summarize(mean.ssr = mean(ssr), sd = sd(ssr)) %>%
  mutate(err_u = mean.ssr + sd/sqrt(3),
         err_l = mean.ssr - sd/sqrt(3)) %>%
  ggplot(., aes(mod, mean.ssr, fill = PM)) +
  geom_col(position = "dodge") +
  geom_errorbar(
    aes(ymax = err_u, ymin = err_l), 
    position = position_dodge(width = .9),
    width = .3) +
  scale_fill_manual(name = "Parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ PM) +
  ggtitle("SSR 2-pool models 0-10 cm") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# SSR, eco
rbind(ssr.2pp.df, ssr.2ps.df) %>%
  mutate(mod = rep(c("2pp", "2ps"), each = nrow(ssr.2pp.df)),
         PM = substr(PMeco_depth, 1, 2),
         eco = substr(PMeco_depth, 3, 4)) %>%
  group_by(eco, mod) %>%
  summarize(mean.ssr = mean(ssr), sd = sd(ssr)) %>%
  mutate(err_u = mean.ssr + sd/sqrt(3),
         err_l = mean.ssr - sd/sqrt(3)) %>%
  ggplot(., aes(mod, mean.ssr, fill = eco)) +
  geom_col(position = "dodge") +
  geom_errorbar(
    aes(ymax = err_u, ymin = err_l), 
    position = position_dodge(width = .9),
    width = .3) +
  facet_wrap(. ~ eco) +
  ggtitle("SSR 2-pool models 0-10 cm (eco)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# var_ms, PM
rbind(var_ms.2pp.df, var_ms.2ps.df) %>%
  mutate(mod = rep(c("2pp", "2ps"), each = nrow(var_ms.2pp.df)),
         PM = substr(PMeco_depth, 1, 2),
         eco = substr(PMeco_depth, 3, 4)) %>%
  group_by(var, PM, mod) %>%
  summarize(mean.var_ms = mean(var_ms), sd = sd(var_ms)) %>%
  mutate(err_u = mean.var_ms + sd/sqrt(3),
         err_l = mean.var_ms - sd/sqrt(3)) %>%
  ggplot(., aes(mod, mean.var_ms, fill = PM)) +
  geom_col(position = "dodge") +
  geom_errorbar(
    aes(ymax = err_u, ymin = err_l), 
    position = position_dodge(width = .9),
    width = .3) +
  scale_fill_manual(name = "Parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ var, scales = "free") +
  ggtitle("Residual error 2-pool models 0-10 cm") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# var_ms, eco
rbind(var_ms.2pp.df, var_ms.2ps.df) %>%
  mutate(mod = rep(c("2pp", "2ps"), each = nrow(var_ms.2pp.df)),
         PM = substr(PMeco_depth, 1, 2),
         eco = substr(PMeco_depth, 3, 4)) %>%
  group_by(var, eco, mod) %>%
  summarize(mean.var_ms = mean(var_ms), sd = sd(var_ms)) %>%
  mutate(err_u = mean.var_ms + sd/sqrt(3),
         err_l = mean.var_ms - sd/sqrt(3)) %>%
  ggplot(., aes(mod, mean.var_ms, fill = eco)) +
  geom_col(position = "dodge") +
  geom_errorbar(
    aes(ymax = err_u, ymin = err_l), 
    position = position_dodge(width = .9),
    width = .3) +
  facet_wrap(. ~ var, scales = "free") +
  ggtitle("Residual error 2-pool models 0-10 cm") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
## System age
# 2pp
SA.2pp.ls <- lapply(seq_along(pars.fit.2pp), function(i) {
  ks <- pars.fit.2pp[[i]][1:2]
  gam <- pars.fit.2pp[[i]][3]
  In <- in.fit.2pp[[i]]
  return(systemAge(, u = In))
})
names(SA.2pp.ls) <- names(pars.fit.2pp)
# 2pp gam = [.5, .95]
SA.2pp.p3.5.95.ls <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  ks <- pars.fit.2pp.p3.5.95[[i]][1:2]
  gam <- pars.fit.2pp.p3.5.95[[i]][3]
  In <- in.fit.2pp.p3.5.95[[i]]
  return(systemAge(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(SA.2pp.p3.5.95.ls) <- names(pars.fit.2pp.p3.5.95)
# 2ps
SA.2ps.ls <- lapply(seq_along(pars.fit.2ps), function(i) {
  ks <- pars.fit.2ps[[i]][1:2]
  gam <- pars.fit.2ps[[i]][3]
  In <- in.fit.2ps[[i]]
  return(systemAge(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(SA.2ps.ls) <- names(pars.fit.2ps)

## Transit time
# 2pp
TT.2pp.ls <- lapply(seq_along(pars.fit.2pp), function(i) {
  ks <- pars.fit.2pp[[i]][1:2]
  gam <- pars.fit.2pp[[i]][3]
  In <- in.fit.2pp[[i]]
  return(transitTime(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(TT.2pp.ls) <- names(pars.fit.2pp)
# 2pp gam = [.5, .95]
TT.2pp.p3.5.95.ls <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  ks <- pars.fit.2pp.p3.5.95[[i]][1:2]
  gam <- pars.fit.2pp.p3.5.95[[i]][3]
  In <- in.fit.2pp.p3.5.95[[i]]
  return(transitTime(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(TT.2pp.p3.5.95.ls) <- names(pars.fit.2pp.p3.5.95)
# 2ps
TT.2ps.ls <- lapply(seq_along(pars.fit.2ps), function(i) {
  ks <- pars.fit.2ps[[i]][1:2]
  gam <- pars.fit.2ps[[i]][3]
  In <- in.fit.2ps[[i]]
  return(transitTime(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(TT.2ps.ls) <- names(pars.fit.2ps)
# compare ages and transit times among the two model structures
SA.2p.ls <- list(SA.2pp.ls, SA.2ps.ls, SA.2pp.p3.5.95.ls)
SA.df <- bind_rows(
  lapply(SA.2p.ls, function(ls) {
    lapply(seq_along(ls), function(i) {
      data.frame(age = c(ls[[i]][["meanSystemAge"]],
                         ls[[i]][["meanPoolAge"]]),
                 component = c("system", "fast pool", "slow pool"))
    })
  })
)
SA.df$PMeco_depth <- rep(names(SA.2pp.ls), each = 3, times = length(SA.2p.ls))
SA.df$Model <- rep(c("2pp", "2ps", "2pp [.5, .95]"), each = 27)
TT.2p.ls <- list(TT.2pp.ls, TT.2ps.ls, TT.2pp.p3.5.95.ls)
TT.df <- bind_rows(
  lapply(TT.2p.ls, function(ls) {
    lapply(seq_along(ls), function(i) {
     data.frame(age = ls[[i]][["meanTransitTime"]],
                component = "transit")
    })
  })
)
TT.df$PMeco_depth <- rep(names(TT.2pp.ls), times = length(TT.2p.ls))
TT.df$Model <- rep(c("2pp", "2ps", "2pp [.5, .95]"), each = 9)
SA.TT.df <- rbind(SA.df, TT.df)
SA.TT.df$PM <- substr(SA.TT.df$PMeco_depth, start = 1, stop = 2)
SA.TT.df$eco <- substr(SA.TT.df$PMeco_depth, start = 3, stop = 4)

## Plot ages and transit times
# by PM
SA.TT.df %>%
  select(!c(PMeco_depth, eco)) %>%
  group_by(component, PM, Model) %>%
  summarize_all(list(mean_age = mean, sd = sd)) %>%
  mutate(err_u = mean_age + sd,
         err_l = mean_age - sd) %>%
  ggplot(., aes(Model, mean_age, fill = PM)) +
  geom_col(position = "dodge") +
  # geom_errorbar(
  #   aes(ymax = err_u, ymin = err_l), 
  #   position = position_dodge(width = .9),
  #   width = .3) +
  scale_fill_manual(name = "Parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ component, scales = "free") +
  ylab("mean age") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# by eco
SA.TT.df %>%
  select(!c(PMeco_depth, PM)) %>%
  group_by(component, eco, Model) %>%
  summarize_all(list(mean_age = mean, sd = sd)) %>%
  mutate(err_u = mean_age + sd,
         err_l = mean_age - sd) %>%
  ggplot(., aes(Model, mean_age, fill = eco)) +
  geom_col(position = "dodge") +
  # geom_errorbar(
  #   aes(ymax = err_u, ymin = err_l),
  #   position = position_dodge(width = .9),
  #   width = .3) +
  facet_wrap(. ~ component, scales = "free") +
  ylab("mean age") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

Bayesian parameter estimation (MCMC)

# the following .RData files are generated by script "sra-ts/source/sra-ts-14c-mcmc-bayes.R"
load(file = "../data/derived/bayes-par-fit-2020-11-06/bayes_fit_2pp_0-10_5000iter.RData")
load(file = "../data/derived/bayes-par-fit-2020-11-17/bayes_fit_2ps_0-10_5000iter.RData")

# # plot parameter convergence
# lapply(bayes_fit_2pp_0_10, plot)
# lapply(bayes_fit_2ps_0_10, plot)

# plot collinearity
iter <- 5000
lapply(bayes_fit_2pp_0_10, pairs, nsample = floor(iter/4))
lapply(bayes_fit_2ps_0_10, pairs, nsample = floor(iter/4))

## look at model performance
pars.bayes.df.fx <- function(mod, pars.bayes, pars.fit) {
  bind_rows(lapply(seq_along(pars.bayes), function(i) {
    ix <- match(unique(pars.bayes[[i]][["pars"]][, 1]), pars.bayes[[i]][["pars"]][, 1])
    df <- data.frame(k1 = pars.bayes[[i]][["pars"]][ix, 1],
                     k2 = pars.bayes[[i]][["pars"]][ix, 2],
                     p3 = pars.bayes[[i]][["pars"]][ix, 3])
    df <- cbind(df,
                PMeco_depth = rep(names(pars.fit)[i], length(ix)),
                mod = rep(mod, length(ix)))
    df <- cbind(df, 
                PM = factor(substr(df$PMeco_depth, 1, 2)),
                eco = factor(substr(df$PMeco_depth, 3, 4), levels = c("pp", "wf", "rf")))
    return(df)
  }))
}
pars.bayes.2pp.df <- pars.bayes.df.fx("2pp", bayes_fit_2pp_0_10, pars.fit.2pp)
pars.bayes.2ps.df <- pars.bayes.df.fx("2ps", bayes_fit_2ps_0_10, pars.fit.2ps)

# # linear fits
# summary(lm(k2 ~ PM, pars.bayes.2pp.df))
# summary(lm(k2 ~ eco, pars.bayes.2pp.df))
# summary(lm(k1 ~ PM, pars.bayes.2pp.df))
# summary(lm(k1 ~ eco, pars.bayes.2pp.df))
# summary(lm(p3 ~ PM, pars.bayes.2pp.df))
# summary(lm(p3 ~ eco, pars.bayes.2pp.df))

# best par set
bestPars.bayes.ls <- lapply(bayes_fit_2pp_0_10, function(x) {
  round(data.frame(k1 = x$bestpar[1],
                   k2 = x$bestpar[2],
                   gam = x$bestpar[3]),
        4)
})
bestPars.bayes.df <- cbind(PM = rep(c("AN", "BS", "GR"), each = 3),
                           eco = rep(c("pp", "rf", "wf"), 3),
                           depth = rep("0-10", 9),
                           bind_rows(bestPars.bayes.ls))

# summarize by PM
pars.bayes.PM <- bestPars.bayes.df %>%
  select(!c(eco, depth)) %>%
  group_by(PM) %>%
  summarize_all(list(mean = mean, sd = sd)) %>%
  mutate_if(is.numeric, format, digits = 3)
# summarize by ECO
pars.bayes.eco <- bestPars.bayes.df %>%
  select(!c(PM, depth)) %>%
  group_by(eco) %>%
  summarize_all(list(mean = mean, sd = sd)) %>%
  mutate_if(is.numeric, format, digits = 3)

# plot best pars
bestPars.bayes.df %>%
  pivot_longer(!(PM:depth), names_to = "par", values_to = "value") %>%
  mutate(PM = factor(PM),
         eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(par, value, color = PM, shape = eco)) +
  geom_jitter(size = 4) +
  scale_color_manual(name = "parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ par, scales = "free") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# plot accepted pars by PM and then by eco
pars.bayes.df %>%
  pivot_longer(!c(PM, eco, PMeco_depth), names_to = "par", values_to = "value") %>%
  mutate(PM = factor(PM),
         eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(par, value, fill = PM)) +
  geom_boxplot() +
  scale_fill_manual(name = "parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ par, scales = "free") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
pars.bayes.df %>%
  pivot_longer(!c(PM, eco, PMeco_depth), names_to = "par", values_to = "value") %>%
  mutate(PM = factor(PM),
         eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(par, value, fill = eco)) +
  geom_boxplot() +
  facet_wrap(. ~ par, scales = "free") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
---
title: "Sierra Nevada Time Series"
author: "J. Beem-Miller"
date: "21 Oct 2020"
output:
  pdf_document:
    latex_engine: xelatex
  html_notebook:
    toc: yes
    toc_depth: 2
    css: custom.css
header_includes:
  - \usepackage[utf8]{inputenc}
  - \usepackage{float}
---
```{r global_options, include = FALSE}
knitr::opts_chunk$set(echo = FALSE, warning = FALSE, message = FALSE,
                      fig.align = 'center', dev = 'cairo_pdf')
```

```{r setup, include = FALSE}
library(ggplot2)
library(dplyr)
# suppress grouping information message
options(dplyr.summarise.inform = FALSE)
library(ISRaD)
library(GSIF)
library(aqp)
library(SoilR)
library(FME)
library(tidyr)
library(readxl)
library(gt)
library(purrr)
```

# Data preparation script for Sierra Nevada time series analysis

```{r load ams-jena-ingest fx}
# 1. Read in isotope data from various sources
# First load helper functions 'read_jena_ams_results.R', 'read_jena_iso_results.R' 
source("./utilities/jena_ams_ingest.R")
source("./utilities/jena_iso_ingest.R")
source("./utilities/jena_elm_ingest.R")
```

```{r plot-funs}
# color palettes for ECO & PM
warm <- "#BF812D"
cool <- "#80CDC1"
cold <- "#01665E"
granite <- "#9daba9"
andesite <- "#382dbf"
basalt <- "#bf382d"
```

```{r read-cn-iso-data, include = FALSE}
# 2. Next read in data from the appropriate directories in 'data/raw'
# 14C
# identify subdirectories in 'raw' directory with "ams_jena" in name
ams_jena_results_dirs <- list.files("../data/raw", pattern = "ams_jena_results", full.names = TRUE)
ams_results_ls <- lapply(seq_along(ams_jena_results_dirs), function(i) {
  read_jena_ams_results(ams_jena_results_dirs[i])
})
names(ams_results_ls) <- list.files("../data/raw", pattern = "ams_jena_results")

# # 13C
# # identify subdirectories in 'raw' directory with "iso_jena" in name
# iso_jena_results_dirs <- list.files("../data/raw", pattern = "iso_jena_results", full.names = TRUE)
# iso_results_ls <- lapply(seq_along(iso_jena_results_dirs), function(i) {
#   read_jena_iso_results(iso_jena_results_dirs[i])
# })
# names(iso_results_ls) <- list.files("../data/raw", pattern = "iso_jena_results")

# Read in C and N data
elm_results_dir <- list.files("../data/raw", pattern = "elm_jena_results", full.names = TRUE)
elm_results_ls <- lapply(seq_along(elm_results_dir), function(i) {
  read_jena_elm_results(elm_results_dir[i])
})
names(elm_results_ls) <- list.files("../data/raw", pattern = "elm_jena_results")
```

```{r data-template}
# Create template for bulk soil data
template19.fx <- function(pm, eco, ndepth) {
  df <- data.frame(Year = rep(2019, ndepth * 3),
                   PM = rep(pm, ndepth * 3),
                   ECO = rep(eco, ndepth * 3),
                   pro_rep = rep(seq(1,3), each = ndepth),
                   lyr_top = rep(seq(0, (ndepth-1) * 10, by = 10), 3),
                   lyr_bot = rep(seq(10, (ndepth) * 10, by = 10), 3))
  df$pro_name <- paste0(df$PM, df$ECO, "_", df$pro_rep)
  df$lyr_name <- paste0(df$pro_name, "_", df$lyr_top, "-", df$lyr_bot)
  return(df)
}

# Create template for composite soil data (incubations, density fractions, etc.)
template.comp.fx <- function(year, pm, eco, depth_bot = c(10, 20, 30), dat) {
  ndepth <- length(depth_bot)
  df <- data.frame(Year = rep(year, ndepth * length(pm)),
                   PM = rep(pm, each = ndepth * length(eco)),
                   ECO = rep(eco, each = ndepth))
  df$lyr_bot <- depth_bot
  df$lyr_top <- sapply(seq_along(depth_bot), function(i) {
    if (i == 1) {
      depth_top <- 0
      } else {
        depth_top <- depth_bot[i - 1]
      }
  })
  df$pro_name <- paste0(df$PM, df$ECO, "_comp")
  n <- nrow(df)
  if (dat == "inc") {
    df <- rbind(df, df)
    df$rep <- rep(c("a", "b"), each = n)
    df$lyr_name <- paste0(df$pro_name, "_", 
                          df$lyr_top, "-", 
                          df$lyr_bot, "_", 
                          df$Year, "_",
                          df$rep)
  } else if (dat == "density") {
    df <- rbind(df, df, df)
    df$frc <- rep(c("fLF", "oLF", "mnC"), each = n)
    df$lyr_name <- paste0(df$pro_name, "_", 
                          df$lyr_top, "-", 
                          df$lyr_bot, "_", 
                          df$Year, "_",
                          df$frc)
  }
  return(df)
}

# templates for bulk soil data
# GRrf 
GRrf <- template19.fx("GR", "rf", 7)
GRrf <- if(any(GRrf$lyr_name == "GRrf_1_60_70")) {
  GRrf <- GRrf[-which(GRrf$lyr_name == "GRrf_1_60_70"), ] # NB: GRrf_1_60_70 doesn't exist
} else {
  GRrf <- GRrf
}
# GRwf
GRwf <- template19.fx("GR", "wf", 9)
# GRpp
GRpp <- template19.fx("GR", "pp", 8)

# ANrf 
ANrf <- template19.fx("AN", "rf", 6)
# ANwf
ANwf <- template19.fx("AN", "wf", 6)
# ANpp
ANpp <- template19.fx("AN", "pp", 8)

# BSrf 
BSrf <- template19.fx("BS", "rf", 8)
BSrf <- if(any(BSrf$lyr_name == "GRrf_1_60_70")) {
  BSrf <- BSrf[-which(BSrf$lyr_name == "BSrf_1_70_80"), ] # NB: BSrf_1_70_80 doesn't exist
} else {
  BSrf <- BSrf
} 
# BSwf
BSwf <- template19.fx("BS", "wf", 7)
# BSpp
BSpp <- template19.fx("BS", "pp", 8)
BSpp[BSpp$lyr_bot == 80, "lyr_bot"] <- 75 # only sampled to 75cm, not 80

sra.2019.df <- rbind(GRrf, GRwf, GRpp,
                     ANrf, ANwf, ANpp,
                     BSrf, BSwf, BSpp)

# template for 2019 incubation data
sra.2019.inc.df <- template.comp.fx(2019, 
                                    pm = c("AN", "BS", "GR"),
                                    eco = c("pp", "wf", "rf"),
                                    dat = "inc")

## template for 2001 incubation data
# list of depths for 2001 inc samples
depth_bot_2001.ls <- list(ANpp = c(6, 13, 33),
                          ANwf = c(11, 35),
                          ANrf = c(11, 32),
                          BSpp = c(7, 18, 28),
                          BSwf = c(10, 19),
                          BSrf = c(8, 15, 30),
                          GRpp = c(7, 15, 27),
                          GRwf = c(4, 13, 28),
                          GRrf = c(8, 27)) 
# template for inputs to template.comp.fx (year, pm, eco)
inc.2001.template <- lapply(seq_along(depth_bot_2001.ls), function(i) {
  nms <- names(depth_bot_2001.ls)
  ls <- list(year = 2001, 
             pm = substr(nms[i], 1, 2), 
             eco = substr(nms[i], 3, 4))
  ls$depth_bot <- depth_bot_2001.ls[[i]]
  return(ls)
})
# create template data frame by iteratively calling template.comp.fx
sra.2001.inc.df <- bind_rows(
  lapply(seq_along(inc.2001.template), function(i) {
    template.comp.fx(year = inc.2001.template[[i]][[1]],
                     pm = inc.2001.template[[i]][[2]],
                     eco = inc.2001.template[[i]][[3]],
                     depth_bot = inc.2001.template[[i]][[4]],
                     dat = "inc")
  })
)

# 2001 bulk soil template
sra.2001 <- vector(mode = "list", length = length(unique(sra.2019.df$pro_name)))
names(sra.2001) <- unique(sra.2019.df$pro_name)

# 2019 bulk soil template
sra.2019 <- sra.2001

# inc templates for merging 14C data
sra.2019.inc <- vector(mode = "list", length = length(unique(sra.2019.inc.df$pro_name)))
names(sra.2019.inc) <- unique(sra.2019.inc.df$pro_name)
sra.2001.inc <- sra.2019.inc
# copies for reps of incubations
sra.2019.inc_L <- sra.2019.inc
names(sra.2019.inc_L) <- substr(names(sra.2019.inc_L), 1, 4)
```

```{r average-cn-data}
# complete cases, convert type for calculating stocks later
# could calculate stocks now and then remove for the following steps where not needed

## 2001 summary data
soc.2001 <- data.frame(read_excel("../data/external/sra_ras_sum/sierra_data_summary_2020.xlsx",
                                  sheet = "2001_bulk_data"))

# create list; remove BS samples deeper than 30 cm
soc.2001.ls <- lapply(split(soc.2001, soc.2001$PMeco), function(df) {
  df <- type.convert(df[complete.cases(df), c("ID", "C.", "mass_kgm2", "PMeco", "pro_rep", "lyr_top", "lyr_bot")])
 return(df[which(df$lyr_bot < 36), ])
})

# Incubation samples combined 0-3 and 3-8 depth increments for BSrf and GRrf
# combine BSrf and GRrf initial depths
# function for calculating weighted average of first two depth increment C content
d1d2.fx <- function(df) {
  d1d2 <- data.frame(ID = paste(df$PMeco[1], df$pro_rep[1], df$lyr_top[1], df$lyr_bot[2], sep = "_"),
                     C. = sum(df$C.[1] * ((df$lyr_bot[1] - df$lyr_top[1]) / df$lyr_bot[2]), df$C.[2] * ((df$lyr_bot[2] - df$lyr_top[2]) / df$lyr_bot[2])),
                     mass_kgm2 = sum(df$mass_kgm2[1], df$mass_kgm2[2]),
                     PMeco = df$PMeco[1],
                     pro_rep = df$pro_rep[1],
                     lyr_top = df$lyr_top[1],
                     lyr_bot = df$lyr_bot[2])
  return(rbind(d1d2,
               df[3:nrow(df), ]))
}
# Run d1d2.fx for BSrf, GRrf
soc.2001.ls.inc <- soc.2001.ls 
soc.2001.ls.inc$GRrf <- bind_rows(lapply(split(soc.2001.ls$GRrf, soc.2001.ls$GRrf$pro_rep), d1d2.fx))

# also need weighted %C for flux weighting
BSrf_comp_01_pctC <- soc.2001.ls.inc$BSrf[which(soc.2001.ls.inc$BSrf$lyr_bot < 9), ]
if (any(c(grepl("BSrf_2_0", BSrf_comp_01_pctC$ID), grepl("BSrf_3_0", BSrf_comp_01_pctC$ID)))) {
  BSrf_comp_01_pctC <- BSrf_comp_01_pctC[-which(BSrf_comp_01_pctC$ID == "BSrf_3_0-3" | BSrf_comp_01_pctC$ID == "BSrf_2_0-3"), ]
}
BSrf_comp_01_pctC$mass_wt <- c(15 / 30, rep((5 / 30), 3))
BSrf_comp_01_pctC$c_pct_wtd <- BSrf_comp_01_pctC$C. * BSrf_comp_01_pctC$mass_wt

# summarize inc SOC
soc.2001.ls.inc <- data.frame(bind_rows(lapply(sra.2001.ls, function(df) {
  df %>%
    mutate(ID2 = paste0(PMeco, "_", lyr_top, "-", lyr_bot)) %>%
    group_by(ID2, PMeco, lyr_top, lyr_bot) %>%
    summarize(c_pct_avg = mean(C.))
})))
if (length(which(soc.2001.sum2$ID2 == "BSrf_0-3" | soc.2001.sum2$ID2 == "BSrf_3-8")) == 2) {
  soc.2001.sum2 <- soc.2001.sum[-which(soc.2001.sum2$ID2 == "BSrf_0-3" | soc.2001.sum2$ID2 == "BSrf_3-8"), ]
  soc.2001.sum2 <- rbind(
    soc.2001.sum2,
    c("BSrf_0-8", "BSrf", 0, 8, sum(BSrf_comp_01_pctC$c_pct_wtd)))
}


# summarize inc
soc.2001.inc.sum <- data.frame(bind_rows(lapply(soc.2001.ls.inc, function(df) {
  df %>%
    mutate(ID2 = paste0(PMeco, "_", lyr_top, "-", lyr_bot)) %>%
    group_by(ID2, PMeco, lyr_top, lyr_bot) %>%
    summarize(c_pct_avg = mean(C.))
})))
save(soc.2001.inc.sum, file = "soc.2001.inc.sum.RData")

# calculate SOC stocks
soc.2001.ls <- lapply(soc.2001.ls, function(df) {
  df$lyr_soc_kgm2 <- df$C. * df$mass_kgm2 * 10^-2
  return(df)
})

# 2019 data
sra.2019.cn.sum <- data.frame(
  bind_rows(unlist(elm_results_ls, recursive = FALSE)) %>%
  mutate(PMeco = sapply(strsplit(ID, "_"), "[", 2),
         depth = sapply(strsplit(ID, "_"), "[", 4)) %>%
  group_by(PMeco, depth) %>%
  summarize(across(c(C, N), .fns = mean))) %>%
  rename(c_pct_avg = C,
         n_pct_avg = N)
sra.2019.cn.sum$ID2 <- paste(sra.2019.cn.sum$PMeco, sra.2019.cn.sum$depth, sep = "_")
save(sra.2019.cn.sum, file = "sra.2019.cn.sum.RData")
```

*Merge templates with 14C, C, and N data*

Radiocarbon analyses for the 2001 samples were not run originally, but were completed on archived samples in 2020.

```{r merge-iso-data-S01-soil}
# Extract 14C data for 2001 samples
ams_results_ls_S01 <- ams_results_ls[grep("S01", names(ams_results_ls))]
for(i in seq_along(sra.2001)) {
  sra.2001[[i]] <- lapply(ams_results_ls_S01, function(ls) {
    lapply(ls, function(df) {
      if(any(grepl(names(sra.2001)[i], df$Probe))) {
       df[grep(names(sra.2001)[i], df$Probe), ] 
      }
    })
  })
  sra.2001[[i]] <- Filter(Negate(is.null), unlist(sra.2001[[i]], recursive = FALSE))
}
sra.2001 <- bind_rows(unlist(sra.2001, recursive = FALSE))

# create ID field, trim df, and add depths
sra.2001$ID <- unlist(strsplit(sra.2001$Probe, "_Sierra Nevada_2001"))
sra.2001 <- sra.2001[ , c("ID", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2001) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
sra.2001$lyr_top <- as.numeric(ifelse(substr(sra.2001$ID, 9, 9) == "-",
                                      substr(sra.2001$ID, 8, 8),
                                      substr(sra.2001$ID, 8, 9)))
sra.2001$lyr_bot <- as.numeric(ifelse(substr(sra.2001$ID, 9, 9) == "-", 
                                      substr(sra.2001$ID, 10, nchar(sra.2001$ID)),
                                      substr(sra.2001$ID, 11, nchar(sra.2001$ID))))
sra.2001$pro_rep <- substr(sra.2001$ID, 6, 6)
sra.2001$PM <- factor(substr(sra.2001$ID, 1, 2))
sra.2001$ECO <- factor(substr(sra.2001$ID, 3, 4), levels = c("pp", "wf", "rf"))
sra.2001$pro_name <- substr(sra.2001$ID, 1, 6)
sra.2001$PMeco <- substr(sra.2001$ID, 1, 4)

# remove outlier ANpp sample
sra.2001 <- sra.2001[-which(sra.2001$ID == "ANpp_3_6-13"), ]

# make list by PMeco
sra.2001.ls <- split(sra.2001, sra.2001$PMeco)
```

```{r merge-iso-data-soil-S19}
# Extract 14C data for 2019 samples
ams_results_ls_S19 <- ams_results_ls[grep("soil-S19", names(ams_results_ls))]
for(i in seq_along(sra.2019)) {
  sra.2019[[i]] <- lapply(ams_results_ls_S19, function(ls) {
    lapply(ls, function(df) {
      if(any(grepl(names(sra.2019)[i], df$Probe))) {
       df[grep(names(sra.2019)[i], df$Probe), ] 
      }
    })
  })
  sra.2019[[i]] <- Filter(Negate(is.null), unlist(sra.2019[[i]], recursive = FALSE))
}
sra.2019 <- bind_rows(unlist(sra.2019, recursive = FALSE))

## merge w/ 2019 template
# rename cols in AMS tables
sra.2019 <- sra.2019[ , c("Probe", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2019) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
# merge
sra.2019.ls <- lapply(split(sra.2019.df, sra.2019.df$lyr_name), function(df) {
  df <- merge(df, sra.2019[grep(df$lyr_name, sra.2019$ID), ])
  df$ID <- NULL
  df$PMeco <- paste0(df$PM, df$ECO)
  return(df)
})

# reshape list by PMeco
sra.2019.ls <- split(bind_rows(sra.2019.ls), bind_rows(sra.2019.ls)[["PMeco"]])
```

```{r merge-iso-data-co2}
### Extract 14C data for incubation samples
## respired CO2, soil
# 2019
ams_results_ls_co2_S19 <- ams_results_ls[grep("co2-S19", names(ams_results_ls))]
for (i in seq_along(sra.2019.inc)) {
  sra.2019.inc[[i]] <- lapply(ams_results_ls_co2_S19, function(ls) {
    lapply(ls, function(df) {
      if (any(grepl(names(sra.2019.inc)[i], df$Probe))) {
        df[grep(names(sra.2019.inc)[i], df$Probe), ] 
      }
    })
  })
  sra.2019.inc[[i]] <- Filter(Negate(is.null), unlist(sra.2019.inc[[i]], recursive = FALSE))
}
sra.2019.inc <- type.convert(
  bind_rows(
    lapply(unlist(sra.2019.inc, recursive = FALSE), 
           function(x) x %>% mutate_all(as.character))),
  as.is = TRUE)
sra.2019.inc <- sra.2019.inc[-which(is.na(sra.2019.inc$F14C)), ]

# 2001
ams_results_ls_co2_S01 <- ams_results_ls[grep("co2-S01", names(ams_results_ls))]
# remove questionable/duplicate samples
# ANrf_comp_11-32_2001_a (analyzed twice; both anomously low compared to rep and rest of data)
ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_22.xlsx` <- ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_22.xlsx`[-grep("ANrf_comp_11-32_2001_a", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_22.xlsx`$Probe), ]
ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-16`$`14C_SierraNevada_Inc_2001_3_results.xlsx` <- ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-16`$`14C_SierraNevada_Inc_2001_3_results.xlsx`[-grep("ANrf_comp_11-32_2001_a", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-16`$`14C_SierraNevada_Inc_2001_3_results.xlsx`$Probe), ]
# from original analysis of samples extracted online 11-Dec-2020 (see readme for notes)
ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_23.xlsx` <- ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2020-12-11`$`Beem-Miller_23.xlsx`[-grep("GRwf_comp_13-28_2001_a_11", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx`$Probe), ]
# from reanalysis of samples extracted online 11-Dec-2020 (see readme for notes)
# GRrf_comp_8-27_2001_a_5, GRrf_comp_8-27_2001_b_6, GRpp_comp_15-27_2001_b_18 
ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx` <- ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx`[c(
  grep("GRrf_comp_8-27_2001_a_5", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx`$Probe),
  grep("GRwf_comp_13-28_2001_b_12", ams_results_ls_co2_S01$`ams_jena_results-co2-S01_2021-01-26`$`14C_SierraNevada_Inc_2001_2_results.xlsx`$Probe)), ]

# create template for extracting data
sra.2001.inc <- vector(mode = "list", length = length(unique(sra.2019.inc.df$pro_name)))
names(sra.2001.inc) <- unique(sra.2019.inc.df$pro_name)
# merge with 14C data
for (i in seq_along(sra.2001.inc)) {
  sra.2001.inc[[i]] <- lapply(ams_results_ls_co2_S01, function(ls) {
    lapply(ls, function(df) {
      if (any(grepl(names(sra.2001.inc)[i], df$Probe))) {
        df[grep(names(sra.2001.inc)[i], df$Probe), ] 
      }
    })
  })
  sra.2001.inc[[i]] <- Filter(Negate(is.null), unlist(sra.2001.inc[[i]], recursive = FALSE))
}
sra.2001.inc <- type.convert(
  bind_rows(
    lapply(unlist(sra.2001.inc, recursive = FALSE), 
           function(x) x %>% mutate_all(as.character))),
  as.is = TRUE)
sra.2001.inc <- sra.2001.inc[-which(is.na(sra.2001.inc$F14C)), ]

# respired CO2, litter
ams_results_ls_co2_L19 <- ams_results_ls[grep("co2-L19", names(ams_results_ls))]
for(i in seq_along(sra.2019.inc_L)) {
  sra.2019.inc_L[[i]] <- lapply(ams_results_ls_co2_L19, function(ls) {
    lapply(ls, function(df) {
      if(any(grepl(names(sra.2019.inc_L)[i], df$Probe))) {
       df[grep(names(sra.2019.inc_L)[i], df$Probe), ] 
      }
    })
  })
  sra.2019.inc_L[[i]] <- Filter(Negate(is.null), unlist(sra.2019.inc_L[[i]], recursive = FALSE))
}
sra.2019.inc_L <- bind_rows(unlist(sra.2019.inc_L, recursive = FALSE))

## merge w/ templates [why do I do this twice?]
# rename cols in AMS tables
# soil CO2
sra.2019.inc <- sra.2019.inc[ , c("Probe", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2019.inc) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
sra.2001.inc <- sra.2001.inc[ , c("Probe", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2001.inc) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
# merge
# 2019
sra.2019.inc.ls <- bind_rows(
  lapply(split(sra.2019.inc.df, sra.2019.inc.df$lyr_name), function(df) {
    df <- merge(df, sra.2019.inc[grep(df$lyr_name, sra.2019.inc$ID), ])
    df$ID <- NULL
    df$PMeco <- paste0(df$PM, df$ECO)
    return(df)
  })
)
sra.2019.inc.ls <- split(sra.2019.inc.ls, sra.2019.inc.ls$PMeco)
# 2001
sra.2001.inc.ls <- bind_rows(
  lapply(split(sra.2001.inc.df, sra.2001.inc.df$lyr_name), function(df) {
    df <- merge(df, sra.2001.inc[grep(df$lyr_name, sra.2001.inc$ID), ])
    df$ID <- NULL
    df$PMeco <- paste0(df$PM, df$ECO)
    return(df)
  })
)
sra.2001.inc.ls <- split(sra.2001.inc.ls, sra.2001.inc.ls$PMeco)

# save inc list
save(sra.2001.inc.ls, file = "sra.2001.inc.ls.RData")

# litter CO2
sra.2019.inc_L <- sra.2019.inc_L[ , c("Probe", "F14C", "err", "∆14C.(‰)", "err.(‰)")]
names(sra.2019.inc_L) <- c("ID", "fm", "fm_err", "d14c", "d14c_err")
sra.2019.inc_L$ID <- substr(substring(sra.2019.inc_L$ID, 
                                      regexpr("_", sra.2019.inc_L$ID) + 1, 
                                      nchar(sra.2019.inc_L$ID)),
                            1, 8)
sra.2019.inc.df_L <- data.frame(Year = rep(2019, 18),
                                rep = rep(c(1, 2), 9),
                                PM = rep(c("AN", "BS", "GR"), each = 6),
                                eco = rep(c("pp", "wf", "rf"), each = 2, times = 3))
sra.2019.inc.df_L$PMeco <- paste0(sra.2019.inc.df_L$PM, sra.2019.inc.df_L$eco)
sra.2019.inc.df_L$ID <- paste0(sra.2019.inc.df_L$PM, sra.2019.inc.df_L$eco, "-L_", sra.2019.inc.df_L$rep)
# add dry wts and litter depth
sra.2019.L <- read.csv("../data/derived/lab_jena_litter/Litter_2019_2021-01-27.csv")
sra.2019.inc.df_L <- merge(sra.2019.inc.df_L, sra.2019.L[ , c("PMeco", "lyr_top", "lyr_bot")], all.x = TRUE)
# merge
sra.2019.inc_L.df <- bind_rows(
  lapply(split(sra.2019.inc_L, sra.2019.inc_L$ID), function(df) {
    df <- merge(df, sra.2019.inc.df_L, by = "ID")
    df$ID <- NULL
    return(df)
  })
)
sra.2019.inc_L.ls <- split(sra.2019.inc_L.df, sra.2019.inc_L.df$PMeco)
```

```{r plot-utils}
# fm and d14c conversion functions
lambda <- 1/8267 # = 1/(true mean life of 14C)
calc_fm <- function(d14c, obs_date_y) {
  ((d14c/1000) + 1)/exp(lambda * (1950 - obs_date_y))
}
calc_14c <- function(fm, obs_date_y) {
  (fm * exp(lambda * (1950 - obs_date_y)) - 1) * 1000
}

# calc atm in 2001, 2009, 2019
Datm <- rbind(graven, future14C)
atm.d14.2001 <- Datm[Datm$Date == 2001.5, "NHc14"]
atm.fm.2001 <- calc_fm(atm.d14.2001, 2001)
atm.d14.2009 <- Datm[Datm$Date == 2009.5, "NHc14"]
atm.fm.2009 <- calc_fm(atm.d14.2009, 2009)
atm.d14.2019 <- Datm[Datm$Date == 2019.5, "NHc14"]
atm.fm.2019 <- calc_fm(atm.d14.2019, 2019)
```

```{r plot-litter-14c}
fig.n <- fig.n + 1
# summarize litter inc data
sra.2019.inc_L.sum <- sra.2019.inc_L.df %>%
  mutate(eco = factor(ifelse(eco == "pp", "warm",
                             ifelse(eco == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = factor(ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))),
         Year = factor("2019")) %>%
  group_by(Year, PMeco, pm, eco, lyr_top, lyr_bot) %>%
  summarize(d14c_mean = mean(d14c),
            d14c_u = max(d14c),
            d14c_l = min(d14c),
            fm_mean = mean(fm),
            fm_u = max(fm),
            fm_l = min(fm))

# plot as cols by climate
sra.2019.inc_L.sum %>%
  mutate(MAT = factor(eco, levels = c("warm", "cool", "cold"), labels = c("10-13", "8-10", "5-6"))) %>%
  ggplot(., aes(MAT, d14c_mean, fill = pm)) +
  geom_hline(yintercept = 0) +
  geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
  geom_col(position = "dodge2") +
  geom_errorbar(aes(ymax = d14c_u, ymin = d14c_l, color = pm), 
                position = position_dodge2(width = .5, padding = .5)) +
  scale_fill_manual(name = "Parent material",
                    values = c("andesite" = "blue", 
                               "basalt" = "red", 
                               "granite" = "darkgray")) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  coord_flip() +
  ylab(expression(Delta*''^14*'C (‰)')) +
  xlab(expression("MAT ("*~degree*C*")")) +
  theme_bw() +
  theme(panel.grid = element_blank())

# plot as points with depth
sra.2019.inc_L.sum %>%
  ggplot(., aes(d14c_mean, lyr_top, color = pm)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_point(size = 3) +
  geom_errorbarh(aes(xmax = d14c_u, xmin = d14c_l), height = 1) +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  facet_grid(rows = vars(eco)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid = element_blank())
```
>**Fig. `r {fig.n}`. Litter incubation $\Delta$^14^C-CO~2~ (2019)**

>*Caption:* Mean $\Delta$^14^C-CO~2~ for each site. Error bars show min and max of duplicate incubation samples. ^a)^ Data shown by site, without litter depth, ^b)^ Data shown by depth of litter layer, binned by climate zone.

```{r plot-14c-profile-fx}
pro.plot <- function(df, maxDepth, min14C, rep) {
  ggplot(df, aes(d14c, lyr_bot, color = PM, shape = ECO, group = rep)) +
    geom_vline(xintercept = 0) +
    geom_hline(yintercept = 0) +
    geom_point(size = 3) +
    geom_path() +
    scale_y_reverse(limits = c(maxDepth, 0)) +
    scale_x_continuous(limits = c(min14C, 180)) +
    scale_color_manual(name = "parent material",
                       labels = c("AN" = "andesite",
                                  "BS" = "basalt",
                                  "GR" = "granite"),
                       values = c("AN" = "blue", 
                                  "BS" = "red", 
                                  "GR" = "darkgray")) +
    scale_shape_manual(name = "ecosystem",
                       labels = c("pp" = expression(italic("P. ponderosa")),
                                  "rf" = expression(italic("A. magnifica")),
                                  "wf" = expression(italic("A. concolor"))),
                       values = c("pp" = 15, 
                                  "rf" = 16, 
                                  "wf" = 17)) +
    xlab(expression(Delta*''^14*'C (‰)')) +
    ylab("Depth (cm)") +
    theme_bw() +
    theme(panel.grid.minor = element_blank())
}
```

```{r plot-2001-profiles}
# lapply(sra.2001.ls, function(df) pro.plot(df, max(df$lyr_bot), min(df$d14c), df$pro_rep))
```

```{r plot-2019-profiles}
# lapply(sra.2019.ls, function(df) pro.plot(df, max(df$lyr_bot), min(df$d14c), df$pro_rep))
```

```{r plot-2019-co2-profiles}
# lapply(sra.2019.inc.ls, function(df) pro.plot(df, max(df$lyr_bot), min(df$d14c), NA))
```

## 2001 mean radiocarbon profiles
```{r plot-2001-avg-profiles}
# combine reps
sra.2001.sum.ls  <- lapply(sra.2001.ls, function(df) {
  df <- data.frame(df %>%
                     filter(lyr_bot <= 40) %>%
                     mutate(lyr_top_ch = as.character(lyr_top),
                            lyr_bot_ch = as.character(lyr_bot)) %>%
                     select(PM, ECO, PMeco, fm, d14c, lyr_top_ch, lyr_bot_ch) %>%
                     group_by(PM, ECO, PMeco, lyr_top_ch, lyr_bot_ch) %>%
                     summarize_all(list(mean = mean, sd = sd), na.rm = TRUE))
  names(df) <- c("PM", "ECO", "PMeco", "lyr_top", "lyr_bot", "fm", "d14c", "fm_sd", "d14c_sd")
  df$lyr_top <- as.numeric(df$lyr_top)
  df$lyr_bot <- as.numeric(df$lyr_bot)
  df$d14c_u <- df$d14c + df$d14c_sd
  df$d14c_l <- df$d14c - df$d14c_sd
  return(df[order(df$lyr_bot), ])
})
sra.01.sum <- bind_rows(sra.2001.sum.ls) %>%
  mutate(eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite")))

# plot
fig.n <- fig.n + 1
sra.01.sum %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, group = PMeco)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path() +
  scale_y_reverse(limits = c(40, 0)) +
  scale_x_continuous(limits = c(-100, 180)) +    
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. Mean profile $\Delta$^14^C for 2001 samples**

>*Caption:* Mean $\Delta$^14^C by depth for each site in 2001. Error bars show ±1 standard deviation, solid vertical line shows $\Delta$^14^C of the atmosphere in the year of sampling.

## 2009 radiocarbon profiles
```{r load-2009-data}
# 2009 summary data (from C. Rasmussen)
ras18.sum <- read_excel(
  "../data/external/sra_ras_sum/sierra_data_summary_2020.xlsx",
  sheet = "Data_Summary_2018_paper")

# remove empty data rows
ras18.sum <- ras18.sum[-which(is.na(ras18.sum$`Sample ID`)), ]

# summarize 09 data
sra.09.sum <- ras18.sum %>%
  mutate(
    ECO = ifelse(Biome == "PP", "pp", ifelse(Biome == "WF", "wf", "rf")),
    PM = ifelse(Parent_Material == "Andesite", "AN", ifelse(Parent_Material == "Basalt", "BS", "GR")),
    eco = factor(ifelse(ECO == "pp", "warm", ifelse(ECO == "wf", "cool", "cold")),
                 levels = c("warm", "cool", "cold")),
    pm = paste0(tolower(substr(Parent_Material, 1, 1)), 
                substr(Parent_Material, 2, nchar(Parent_Material))),
    mass_kgm2 = BD_g_cm_3 * Soil_finefraction * (`bottom mineral` - `top mineral`) * 10) %>%
  mutate(PMeco = paste0(PM, ECO)) %>%
  rename(d14c = "Δ14C",
         lyr_bot = "bottom mineral",
         lyr_top = "top mineral")
sra.2009.ls <- lapply(split(sra.09.sum, sra.09.sum$PMeco), function(df) {
  df$lyr_fraction_modern <- calc_fm(df$d14c, 2009)
  return(data.frame(df))
})

# 2009 bulk C data
ras18.blkC <- as.data.frame(read_excel(
  "../data/external/sra_ras_sum/sierra_data_summary_2020.xlsx",
  sheet = "2009_bulk_data"))

# Add PM, ECO, mass_kgm2 vars
ras18.blkC$ECO <- ifelse(ras18.blkC$Biome == "PP", "pp", ifelse(ras18.blkC$Biome == "RF", "rf", "wf"))
ras18.blkC$PMeco <- paste0(ras18.blkC$PM, ras18.blkC$ECO)
ras18.blkC$mass_kgm2 <- ras18.blkC$Thickness_cm * ras18.blkC$BD_g_cm_3 * ras18.blkC$Soil_finefraction * 10

# Calculate SOC stocks
ras18.blkC$lyr_soc <- ras18.blkC$Thickness_cm * ras18.blkC$BD_g_cm_3 * ras18.blkC$Soil_finefraction * ras18.blkC$C_pct * 10^-1

# Calculate cmtv SOC stocks
ras18.blkC$lyr_soc_cmtv <- unlist(lapply(split(ras18.blkC, ras18.blkC$pro_name), function(x) {
  x$lyr_soc_cmtv <- NA
  for (i in seq(nrow(x))) {
    if (i == 1) {
      x$lyr_soc_cmtv[i] <- x$lyr_soc[i]
    } else {
      x$lyr_soc_cmtv[i] <- x$lyr_soc[i] + x$lyr_soc_cmtv[i - 1]
    }
  }
  return(x$lyr_soc_cmtv)
}))
```

```{r error-BD-pred}
# LOOCV function, fit = lm mod
loocv <- function (fit) {
  h <- lm.influence(fit)$h
  mean((residuals(fit) / (1-h))^2)
}

# test function for predicting BD as function of PM, ECO, and C content
bd.mod <- lm(BD_g_cm_3 ~ PM * ECO + PM * C_pct + `bottom mineral`, ras18.blkC)
bd.pred <- predict.lm(bd.mod, ras18.blkC, interval = "predict", pred.var = loocv(bd.mod))
bd.err.df <- ras18.blkC
bd.err.df$BD_pred <- bd.pred[ , 1]
bd.err.df$BD_l <- bd.pred[ , 2]
bd.err.df$BD_u <- bd.pred[ , 3]

# plot
ggplot(bd.err.df, aes(BD_g_cm_3, BD_pred)) +
  geom_ribbon(aes(ymin = BD_l, ymax = BD_u), fill = "lightgray") +
  geom_smooth(method = "lm", formula = y ~ x, se = FALSE, color = "black") +
  geom_point(aes(color = PM, shape = ECO, size = `bottom mineral`/10)) +
  scale_color_manual(name = "Parent material",
                     values = c("AN" = andesite,
                                "BS" = basalt,
                                "GR" = granite)) +
  theme_bw() +
  theme(panel.grid = element_blank())
```

```{r plot-2009-14c-profiles}
# plot
fig.n <- fig.n + 1
sra.09.sum %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, group = PMeco)) +
  geom_vline(xintercept = atm.d14.2009) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_path(linetype = "dashed") +
  scale_y_reverse() +
  scale_x_continuous(limits = c(-100, 180)) +    
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "ecosystem",
                     values = c("warm" = 0, 
                                "cool" = 1, 
                                "cold" = 2)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. Profile $\Delta$^14^C for 2009 samples**

>*Caption:* Profile $\Delta$^14^C by depth for each site in 2009. Solid vertical line shows $\Delta$^14^C of the atmosphere in the year of sampling. Error bars not shown as only a single replicate profile was analyzed per site.

## 2019 mean radiocarbon profiles
```{r plot-2019-avg-profiles}
# combine reps
sra.2019.sum.ls  <- lapply(sra.2019.ls, function(df) {
  df <- data.frame(df %>%
                     mutate(lyr_top_ch = as.character(lyr_top),
                            lyr_bot_ch = as.character(lyr_bot)) %>%
                     select(PM, ECO, PMeco, fm, d14c, lyr_top_ch, lyr_bot_ch) %>%
                     group_by(PM, ECO, PMeco, lyr_top_ch, lyr_bot_ch) %>%
                     summarize_all(list(mean = mean, sd = sd), na.rm = TRUE))
  names(df) <- c("PM", "ECO", "PMeco", "lyr_top", "lyr_bot", "fm", "d14c", "fm_sd", "d14c_sd")
  df$lyr_top <- as.numeric(df$lyr_top)
  df$lyr_bot <- as.numeric(df$lyr_bot)
  df$d14c_u <- df$d14c + df$d14c_sd
  df$d14c_l <- df$d14c - df$d14c_sd
  return(df)
})
sra.19.sum <- bind_rows(sra.2019.sum.ls) %>%
  mutate(eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) 

# plot
fig.n <- fig.n + 1
sra.19.sum %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, group = PMeco)) +
  geom_vline(xintercept = atm.d14.2019) +
  geom_hline(yintercept = 0) +
  geom_point(size = 2.7) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path() +
  scale_y_reverse(limits = c(max(sra.19.sum$lyr_bot), 0)) +
  scale_x_continuous(limits = c(min(sra.19.sum$d14c), 180)) +    
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. Mean profile $\Delta$^14^C for 2019 samples**

>*Caption:* Mean $\Delta$^14^C by depth for each site in 2019. Error bars show ±1 standard deviation, solid vertical line shows $\Delta$^14^C of the atmosphere in the year of sampling.

## Change in $\Delta$^14^C over time between 2001 and 2019 

```{r plot-all-avg}
# combine '01 and '19 data for plotting
sra.01.sum$Year <- 2001
sra.19.sum$Year <- 2019

sra.01.19.sum <- rbind(sra.01.sum, sra.19.sum)
sra.01.19.sum$Year <- as.factor(sra.01.19.sum$Year)

fig.n <- fig.n + 1
sra.01.19.sum %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         ecoYear = paste0(ECO, Year),
         ecoYear2 = ifelse(ecoYear == "pp2001", "warm (2001)",
                           ifelse(ecoYear == "pp2019", "warm (2019)",
                                  ifelse(ecoYear == "wf2001", "cool (2001)",
                                         ifelse(ecoYear == "wf2019", "cool (2019)",
                                                ifelse(ecoYear == "rf2001", "cold (2001)", "cold (2019)")))))) %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = ecoYear2, group = PMeco_year)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path(aes(linetype = Year)) +
  scale_y_reverse(limits = c(41, 0)) +
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. Mean profile $\Delta$^14^C for 2001 and 2019 samples**

>*Caption:* Mean $\Delta$^14^C by depth for each site in 2001 and 2019. Error bars show ±1 standard deviation. Vertical lines show $\Delta$^14^C of the atmosphere in 2001 (solid) and 2019 (dashed).

## Incubation $\Delta$^14^C-CO~2~

```{r inc-d14c-plot-setup}
## 2019
sra.2019.inc.df <- bind_rows(sra.2019.inc.ls)
# add litter inc data and summarize
sra.2019.inc.sum.df <- data.frame(rbind(
  sra.2019.inc_L.sum,
  sra.2019.inc.df %>%
    mutate(eco = factor(ifelse(ECO == "pp", "warm",
                             ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
           pm = factor(ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))),
           # remove GRrf 10-20 "a" point
           d14c = replace(d14c, which(d14c < -300), NA),
           Year = factor(Year)) %>%
  group_by(Year, PMeco, pm, eco, lyr_bot, lyr_top) %>%
  summarize(d14c_mean = mean(d14c, na.rm = TRUE),
            d14c_l = min(d14c, na.rm = TRUE),
            d14c_u = max(d14c, na.rm = TRUE),
            fm_mean = mean(fm),
            fm_l = min(fm),
            fm_u = max(fm))))
sra.2019.inc.sum.ls <- split(sra.2019.inc.sum.df, sra.2019.inc.sum.df$PMeco)

# 2001
sra.2001.inc.df <- bind_rows(sra.2001.inc.ls)
sra.2001.inc.sum.df <- data.frame(
  sra.2001.inc.df %>%
    mutate(eco = factor(ifelse(ECO == "pp", "warm",
                               ifelse(ECO == "wf", "cool", "cold")),
                        levels = c("warm", "cool", "cold")),
           pm = factor(ifelse(PM == "AN", "andesite",
                       ifelse(PM == "BS", "basalt", "granite"))),
           Year = factor(Year)) %>%
    group_by(Year, PMeco, pm, eco, lyr_bot, lyr_top) %>%
    summarize(d14c_mean = mean(d14c),
              d14c_l = min(d14c),
              d14c_u = max(d14c),
              fm_mean = mean(fm),
              fm_l = min(fm),
              fm_u = max(fm))
)
sra.2001.inc.sum.ls <- split(sra.2001.inc.sum.df, sra.2001.inc.sum.df$PMeco)
sra.2001.inc.sum.df <- sra.2001.inc.sum.df[ , !(names(sra.2001.inc.sum.df) %in% c("fm_mean", "fm_l", "fm_u", "lyr_top", "PMeco"))]
```

```{r plot-inc-d14c-2019}
# 2019
fig.n <- fig.n + 1
sra.2019.inc.sum.df[order(sra.2019.inc.sum.df$lyr_bot), ] %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = eco)) +
  geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l,
        xmax = d14c_u,
        color = pm),
    height = 1.5) +
  geom_path(linetype = "dashed") +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 0, 
                                "cool" = 1, 
                                "cold" = 2)) +
  xlab(expression('Incubation '*Delta*''^14*'C-CO'[2]*' (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. $\Delta$^14^C-CO~2~ of 2019 bulk soil incubations**

>*Caption:* $\Delta$^14^CO~2~ by depth for each site in 2019. One rep from GRrf 10-20 (the 10-20 cm increment sample from the cold granite site) is strongly depleted relative to the other rep: $\Delta$^14^C-CO~2~ = `r {sra.2019.inc.df[sra.2019.inc.df$PMeco == "GRrf" & sra.2019.inc.df$lyr_bot == 20, "d14c"]}`. The highly depleted sample has been excluded for display reasons.

```{r plot-inc-d14c-2001}
# plot 2001 data
fig.n <- fig.n + 1
sra.2001.inc.sum.df %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = eco)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l,
        xmax = d14c_u,
        color = pm),
    height = 1.5) +
  geom_path() +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue",
                                "basalt" = "red",
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15,
                                "cool" = 16,
                                "cold" = 17)) +
  scale_x_continuous(limits = c(-70, 190)) +
  xlab(expression('Incubation '*Delta*''^14*'C-CO'[2]*' (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. $\Delta$^14^C-CO~2~ of 2001 bulk soil incubations**

>*Caption:* $\Delta$^14^CO~2~ by depth for each site in 2001. Note that some sites only have two depth increments. Similar to the 2019 dataset, one of the GRrf reps from the deepest depth increment was strongly depleted: $\Delta$^14^C-CO~2~ = `r {sra.2001.inc.sum.df %>% filter(pm == "granite", eco == "cold", lyr_bot == 27) %>% pull("d14c_l", "d14c_u")}`. Both points have been excluded for display reasons.

```{r plot-inc-d14c-all}
# plot together
sra.inc.all <- rbind(sra.2001.inc.sum.df, 
                     sra.2019.inc.sum.df[ , names(sra.2019.inc.sum.df) %in% names(sra.2001.inc.sum.df)])
save(sra.inc.all, file = "sra.inc.all.RData")

fig.n <- fig.n + 1 
sra.inc.all %>%
  filter(lyr_bot > 0) %>%
  mutate(PMeco_year = paste0(pm, eco, Year),
         ecoYear = paste0(eco, " (", Year, ")")) %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = ecoYear, group = PMeco_year)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path(aes(linetype = Year)) +
  scale_y_reverse(limits = c(41, 0)) +
  scale_color_manual(name = "parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_x_continuous(limits = c(-70, 190)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. $\Delta$^14^C-CO~2~ of 2001 and 2019 bulk soil incubations**

>*Caption:* $\Delta$^14^CO~2~ by depth for each site in 2001 and 2019. Different depth increments were sampled in 2001 and 2019. Points are the mean of laboratory duplicates; error bars are the measured values of each duplicate. Granite/cold point exlcuded for display reasons as it is strongly depleted.

## Incubation vs. bulk soil $\Delta$^14^C

```{r inc-bulk-d14c-plot-setup}
# bind rows of inc list
sra.19.inc <- sra.2019.inc.sum.df
sra.19.inc$Type <- "inc"

# 2001
sra.01.inc <- sra.2001.inc.sum.df
sra.01.inc$Type <- "inc"

# rbind bulk data
sra.19.bulk <- sra.19.sum[which(sra.19.sum$lyr_bot < 31), c("Year", "PM", "ECO", "lyr_bot","d14c", "d14c_l", "d14c_u")]
names(sra.19.bulk)[which(names(sra.19.bulk) == "d14c")] <- "d14c_mean"
sra.19.bulk$Type <- "bulk"
sra.19.bulk <- sra.19.bulk %>%
  mutate(pm = factor(ifelse(PM == "AN", "andesite",
                            ifelse(PM == "BS", "basalt", "granite"))),
         eco = factor(ifelse(ECO == "pp", "warm",
                             ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         PM = NULL,
         ECO = NULL)
sra.19.inc.blk <- rbind(sra.19.bulk, sra.19.inc[ , names(sra.19.inc) %in% names(sra.19.bulk)])
save(sra.19.inc.blk, file = "sra.19.inc.blk.RData")

# 2001
# Need to calculate weighted average of radiocarbon values and stocks for combined inc depths 
# 1) add SOC stocks to duplicate sra.2001.ls obj
sra.2001.ls2 <- sra.2001.ls
for(i in seq_along(sra.2001.ls2)) {
  ix <- match(sra.2001.ls2[[i]][["ID"]], soc.2001.ls[[i]][["ID"]])
  sra.2001.ls2[[i]]["lyr_soc_kgm2"] <- soc.2001.ls[[i]][ix, "lyr_soc_kgm2"]
}
# 2) weighted average fx
d1d2.14c.fx <- function(df) {
  sum_soc <- sum(df[1:2, "lyr_soc_kgm2"])
  wt1 <- df$lyr_soc_kgm2[1] / sum_soc
  wt2 <- df$lyr_soc_kgm2[2] / sum_soc
  d1d2 <- df[1, ]
  d1d2$ID = paste(df$PMeco[1], df$pro_rep[1], df$lyr_top[1], df$lyr_bot[2], sep = "_")
  d1d2$lyr_soc_kgm2 = sum(df$lyr_soc_kgm2[1], df$lyr_soc_kgm2[2])
  d1d2$lyr_bot = df$lyr_bot[2]
  d1d2$fm <- sum(df$fm[1] * wt1, df$fm[2] * wt2)
  d1d2$d14c <- sum(df$d14c[1] * wt1, df$d14c[2] * wt2)
  return(rbind(d1d2,
               df[3:nrow(df), ]))
}
# 3) calc. wtd. average for GRrf
sra.2001.ls2$GRrf <- bind_rows(
  lapply(split(sra.2001.ls2$GRrf, sra.2001.ls2$GRrf$pro_rep), function(x) {
    d1d2.14c.fx(x)
  })
)
# 4) calc. wtd. average for BSrf
#    - problem here is that only one pro_rep has 0-3 cm data
#    - so, need to calculate weighted SOC, then calculate weighted 14C
#    - composite 0-8 = 15g BSrf_1_0-3 + 5 g from each pro_rep BSrf_3-8
BSrf_comp_01_i <- sra.2001.ls2$BSrf[which(sra.2001.ls2$BSrf$lyr_bot < 9), ]
BSrf_comp_01_i$soc_wt <- c(15 / 30, rep(5 / 30, 3))
BSrf_comp_01_i$soc_wtd <- BSrf_comp_01_i$lyr_soc_kgm2 * BSrf_comp_01_i$soc_wt

# create summarized list
sra.2001.sum.ls2  <- lapply(sra.2001.ls2, function(df) {
  data.frame(
    df %>%
      filter(lyr_bot <= 40) %>%
      mutate(lyr_bot_ch = as.character(lyr_bot)) %>%
      select(PMeco, d14c, fm, lyr_bot_ch, lyr_soc_kgm2) %>%
      group_by(PMeco, lyr_bot_ch) %>%
      summarize(
        across(where(is.numeric), list(mean = mean, sd = sd), na.rm = TRUE)) %>%
      mutate(lyr_bot = as.numeric(lyr_bot_ch)) %>%
      select(-lyr_bot_ch)
  )
})

# remove BSrf row w/ lyr_bot = 3
sra.2001.sum.ls2$BSrf <- sra.2001.sum.ls2$BSrf[-which(sra.2001.sum.ls2$BSrf$lyr_bot == 3), ]
# calculate weighted average for d14c, fm, lyr_soc_kgm2
sra.2001.sum.ls2$BSrf[which(sra.2001.sum.ls2$BSrf$lyr_bot == 8), "d14c_mean"] <- sum(BSrf_comp_01_i$d14c * (BSrf_comp_01_i$soc_wtd / sum(BSrf_comp_01_i$soc_wtd)))
sra.2001.sum.ls2$BSrf[which(sra.2001.sum.ls2$BSrf$lyr_bot == 8), "fm_mean"] <- sum(BSrf_comp_01_i$fm * (BSrf_comp_01_i$soc_wtd / sum(BSrf_comp_01_i$soc_wtd)))
sra.2001.sum.ls2$BSrf[which(sra.2001.sum.ls2$BSrf$lyr_bot == 8), "lyr_soc_kgm2_mean"] <- sum(BSrf_comp_01_i$soc_wtd)
sra.2001.sum.ls2$BSrf[which(sra.2001.sum.ls2$BSrf$lyr_bot == 8), c("d14c_sd", "fm_sd", "lyr_soc_kgm2_sd")] <- NA

# calculate cmtv soc
sra.2001.sum.ls2 <- lapply(sra.2001.sum.ls2, function(x) {
  x <- x[order(x$lyr_bot), ]
  x$lyr_soc_cmtv <- NA
  for(i in seq_along(x$lyr_bot)) {
      if(i == 1) {
        x$lyr_soc_cmtv[i] <- x$lyr_soc_kgm2_mean[i]
      } else {
        x$lyr_soc_cmtv[i] <- x$lyr_soc_kgm2_mean[i] + x$lyr_soc_cmtv[i-1] 
      }
  }
  return(x)
})

# make df
sra.01.sum <- data.frame(bind_rows(
  lapply(sra.2001.sum.ls2, function(df) {
    df %>%
      mutate(eco = factor(ifelse(grepl("pp", df$PMeco), "warm",
                                 ifelse(grepl("wf", df$PMeco), "cool", "cold")),
                          levels = c("warm", "cool", "cold")),
             pm = ifelse(grepl("AN", df$PMeco), "andesite",
                         ifelse(grepl("BS", df$PMeco), "basalt", "granite")),
             d14c_u = d14c_mean + d14c_sd,
             d14c_l = d14c_mean - d14c_sd,
             Year = 2001,
             Type = "bulk") %>%
      select(names(sra.01.inc)) %>%
      arrange(lyr_bot)
  })
))
# bind with inc
sra.01.inc.blk <- rbind(data.frame(sra.01.inc), sra.01.sum)
save(sra.01.inc.blk, file = "sra.01.inc.blk.RData")
```

```{r read-resp-ts}
## read in timeseries of CO2 release from incubations
# 2019
sra.19a.co2.ts <- read.csv("../data/derived/lab_jena_CO2-timeseries/S19a_CO2_flux_2021-01-19.csv")
sra.19b.co2.ts <- read.csv("../data/derived/lab_jena_CO2-timeseries/S19b_CO2_flux_2021-01-19.csv")

# 2001
sra.01.1.co2.ts <- read.csv("../data/derived/lab_jena_CO2-timeseries/S01_1_CO2_flux_2021-01-27.csv")
sra.01.2.co2.ts <- read.csv("../data/derived/lab_jena_CO2-timeseries/S01_2_CO2_flux_2021-01-27.csv")

## Test that required names are present
nms <- c("PMeco", "ID", "dw_g", "timepoint_cmtv",  "time_d", "mgCO2_jar")
invisible(lapply(list(sra.19a.co2.ts,
                      sra.19b.co2.ts,
                      sra.01.1.co2.ts,
                      sra.01.2.co2.ts),
       function(x) {
         ifelse(!is.na(match(nms, names(x))), "yes", "no")
       }
       ))

# combine all data, remove time points without CO2 measurements, and add year and rep 
ts <- bind_rows(sra.19a.co2.ts[ , nms], 
                sra.19b.co2.ts[ , nms], 
                sra.01.1.co2.ts[ , nms],
                sra.01.2.co2.ts[ , nms])
if(length(which(is.na(ts$mgCO2_jar))) > 0) {
  ts <- ts[-which(is.na(ts$mgCO2_jar)), ]
}
ts$year <- sapply(strsplit(as.character(ts$ID), "_"), "[[", 3)
ts$rep <- sapply(strsplit(as.character(ts$ID), "_"), "[[", 4)
ts$depth <- sapply(strsplit(as.character(ts$ID), "_"), "[[", 2)
ts$ID2 <- paste(ts$PMeco, ts$depth, sep = "_")

# add C content
ts[which(ts$year == 2001), "gC_gS"] <- soc.2001.sum2[match(ts[which(ts$year == 2001), "ID2"], soc.2001.sum2$ID2), "c_pct_avg"] * 10^-2
ts[which(ts$year == 2019), "gC_gS"] <- sra.2019.cn.sum[match(ts[which(ts$year == 2019), "ID2"], sra.2019.cn.sum$ID2), "c_pct_avg"] * 10^-2

# calculate per unit carbon fluxes
ts$mgCO2_gC <- ts$gC_gS * ts$dw_g * ts$mgCO2_jar * (12/44)
ts$mgCO2_gC_d <- ts$mgCO2_gC / ts$time_d

# average reps
ts.avg <- ts %>%
  group_by(PMeco, year, depth, timepoint_cmtv) %>%
  summarize(time_d = mean(time_d),
            mgCO2_gC_d_avg = mean(mgCO2_gC_d),
            mgCO2_gC_d_max = max(mgCO2_gC_d),
            mgCO2_gC_d_min = min(mgCO2_gC_d),
            mgCO2_gC_avg = mean(mgCO2_gC),
            mgCO2_gC_max = max(mgCO2_gC),
            mgCO2_gC_min = min(mgCO2_gC)) %>%
  mutate(PMeco_depth_year = paste(PMeco, depth, year, sep = "_"))

# add depth index
t1 <- ts.avg[ts.avg$timepoint_cmtv == 1, ]
t1 <- data.frame(
  bind_rows(
    lapply(split(t1, t1$year), function(df) {
      bind_rows(lapply(split(df, df$PMeco), function(x) {
        x$lyr_top <- as.numeric(sapply(strsplit(x$depth, "-"), "[", 1))
        x <- x[order(x$lyr_top), ]
        x$depth_index <- seq(1, nrow(x))
        return(x)
      }))
    })))
ts.avg$depth_index <- t1[match(ts.avg$PMeco_depth_year, t1$PMeco_depth_year), "depth_index"]

# cumulative flux rates
ts.avg.cmtv <- bind_rows(
  lapply(split(ts.avg, ts.avg$PMeco_depth_year), function(x) x[nrow(x), ]), 
  .id = "PMeco_depth_year")
```

```{r plot-resp-rates}
fig.n <- 1
# function for plotting
ts.plot.fx <- function(df, yr, increment, cumulative = TRUE) {
      if (cumulative) {
        df %>%
          filter(year == yr & depth_index == increment) %>%
          mutate(PM = ifelse(grepl("AN", PMeco), "AN",
                             ifelse(grepl("BS", PMeco), "BS", "GR")),
                 eco = factor(ifelse(grepl("rf", PMeco), "rf", 
                                     ifelse(grepl("wf", PMeco), "wf", "pp")),
                              levels = c("pp", "wf", "rf"))) %>%
          ggplot(., aes(time_d, mgCO2_gC_avg, color = PM, shape = eco)) +
          geom_ribbon(aes(ymin = mgCO2_gC_max, ymax = mgCO2_gC_min, fill = PM, linetype = eco, alpha = 0.2), show.legend = FALSE) +
          geom_point(aes(time_d, mgCO2_gC_max, fill = PM, shape = eco), color = "black", size = 3, stroke = 1) +
          geom_point(aes(time_d, mgCO2_gC_min, fill = PM, shape = eco), color = "black", size = 3, stroke = 1) +
          geom_line(aes(color = PM, linetype = eco), size = 1.2) +
          facet_grid(rows = vars(eco),
                     labeller = labeller(eco = c("rf" = "cold", "wf" = "cool", "pp" = "warm"))) +
          scale_x_continuous(limits = c(0,30)) +
          scale_color_manual(name = "Parent material",
                             labels = c("AN" = "andesite",
                                        "BS" = "basalt",
                                        "GR" = "granite"),
                             values = c("AN" = "blue", 
                                        "BS" = "red", 
                                        "GR" = "darkgray")) +
          scale_shape_manual(name = "Climate",
                             labels = c("rf" = "cold",
                                        "wf" = "cool",
                                        "pp" = "warm"),
                             values = c("pp" = 21, 
                                        "rf" = 22, 
                                        "wf" = 24)) +
          scale_fill_manual(values =c("AN" = "blue",
                                      "BS" = "red",
                                      "GR" = "darkgray")) +
          scale_linetype_manual(name = "Climate",
                                values = c("rf" = "dotted",
                                           "wf" = "dashed",
                                           "pp" = "solid"),
                                labels = c("rf" = "cold",
                                        "wf" = "cool",
                                        "pp" = "warm")) +
          ylab(expression('Cumulative flux (mgCO'[2]*'-C gC'^-1*')')) +
          xlab("Time (days)") +
          guides(color = guide_legend(order = 1),
                 shape = guide_legend(order = 3)) +
          ggtitle(paste("Cumulative flux, ", yr, "depth ", increment)) +
          theme_bw() +
          theme(panel.grid = element_blank())
    } else {
       df %>%
        filter(year == yr & depth_index == increment) %>%
        mutate(PM = ifelse(grepl("AN", PMeco), "AN",
                           ifelse(grepl("BS", PMeco), "BS", "GR")),
              eco = factor(ifelse(grepl("rf", PMeco), "rf",
                                  ifelse(grepl("wf", PMeco), "wf", "pp")),
                           levels = c("pp", "wf", "rf"))) %>%
        ggplot(., aes(time_d, mgCO2_gC_d_avg, color = PM, shape = eco)) +
        geom_ribbon(aes(ymin = mgCO2_gC_d_max, ymax = mgCO2_gC_d_min, fill = PM, linetype = eco, alpha = 0.2), show.legend = FALSE) +
          geom_point(aes(time_d, mgCO2_gC_d_max, fill = PM, shape = eco), color = "black", size = 3, stroke = 1) +
          geom_point(aes(time_d, mgCO2_gC_d_min, fill = PM, shape = eco), color = "black", size = 3, stroke = 1) +
        geom_line(aes(color = PM, linetype = eco), size = 1.2) +
        facet_grid(rows = vars(eco),
                   labeller = labeller(eco = c("rf" = "cold", "wf" = "cool", "pp" = "warm"))) +
        scale_x_continuous(limits = c(0,30)) +
        scale_color_manual(name = "Parent material",
                           labels = c("AN" = "andesite",
                                      "BS" = "basalt",
                                      "GR" = "granite"),
                           values = c("AN" = "blue", 
                                      "BS" = "red", 
                                      "GR" = "darkgray")) +
        scale_shape_manual(name = "Climate",
                           labels = c("rf" = "cold",
                                      "wf" = "cool",
                                      "pp" = "warm"),
                           values = c("pp" = 21, 
                                      "rf" = 22, 
                                      "wf" = 24)) +
        scale_fill_manual(values =c("AN" = "blue",
                                    "BS" = "red",
                                    "GR" = "darkgray")) +
        scale_linetype_manual(name = "Climate",
                              values = c("rf" = "dotted",
                                         "wf" = "dashed",
                                         "pp" = "solid"),
                              labels = c("rf" = "cold",
                                      "wf" = "cool",
                                      "pp" = "warm")) +
        ylab(expression('Respiration Rate (mgCO'[2]*'-C gC'^-1*'d'^-1*')')) +
        xlab("Time (days)") +
        guides(color = guide_legend(order = 1),
               shape = guide_legend(order = 3)) +
        ggtitle(paste("Flux rate", yr, "depth ", increment)) +
        theme_bw() +
        theme(panel.grid = element_blank())
    }
}

## cumulative flux
# 2019
ts.plot.fx(ts.avg, yr = "2019", increment = "1")
ts.plot.fx(ts.avg, yr = "2019", increment = "2")
ts.plot.fx(ts.avg, yr = "2019", increment = "3")
# 2001
ts.plot.fx(ts.avg, yr = "2001", increment = "1")
ts.plot.fx(ts.avg, yr = "2001", increment = "2")
ts.plot.fx(ts.avg, yr = "2001", increment = "3")

## flux rates
# 2019
ts.plot.fx(ts.avg, yr = "2019", increment = "1", cumulative = FALSE)
ts.plot.fx(ts.avg, yr = "2019", increment = "2", cumulative = FALSE)
ts.plot.fx(ts.avg, yr = "2019", increment = "3", cumulative = FALSE)
# 2001
ts.plot.fx(ts.avg, yr = "2001", increment = "1", cumulative = FALSE)
ts.plot.fx(ts.avg, yr = "2001", increment = "2", cumulative = FALSE)
ts.plot.fx(ts.avg, yr = "2001", increment = "3", cumulative = FALSE)
```
>**Fig. `r {fig.n}`. Respiration data from incubations of 2019 and 2001 bulk soils. **

>*Caption:* Points show measured CO~2~ production of laboratory duplicates as cumulative fluxes or daily flux rates by depth, lines show the means, and the ribbon represents the range. 

```{r plot-inc-blk-2019}
# plot 2019
fig.n <- fig.n + 1
# p <-
sra.19.inc.blk %>%
  mutate(ECOtype = paste0(eco, " (", Type, ")")) %>%
  arrange(lyr_bot) %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = ECOtype, linetype = Type)) +
  geom_vline(xintercept = atm.d14.2019) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path() +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem (type)",
                     values = c("warm (bulk)" = 15, 
                                "cool (bulk)" = 16, 
                                "cold (bulk)" = 17,
                                "warm (inc)" = 0, 
                                "cool (inc)" = 1, 
                                "cold (inc)" = 2)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# ggsave("sra.bulkInc.19.pdf", p, device = cairo_pdf, width = 9.5, height = 5, units = "in")
```
>**Fig. `r {fig.n}`. $\Delta$^14^C of 2019 bulk soil incubations and corresponding bulk soil**

>*Caption:* $\Delta$^14^C of bulk soil and respired CO~2~ by depth for each site in 2019. Error bars show one standard deviation for bulk soil, points show mean of three replicate profiles for bulk soils and single observations for respired CO~2~. 

```{r plot-inc-blk-2001}
# plot 2001
fig.n <- fig.n + 1
sra.01.inc.blk %>%
  mutate(ECOtype = paste0(eco, " (", Type, ")")) %>%
  ggplot(., aes(d14c_mean, lyr_bot, color = pm, shape = ECOtype, linetype = Type)) +
  geom_vline(xintercept = atm.d14.2001) +
  geom_hline(yintercept = 0) +
  geom_point(size = 3) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  geom_path() +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem (type)",
                     values = c("warm (bulk)" = 15, 
                                "cool (bulk)" = 16, 
                                "cold (bulk)" = 17,
                                "warm (inc)" = 0, 
                                "cool (inc)" = 1, 
                                "cold (inc)" = 2)) +
  scale_x_continuous(limits = c(-100, 200)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. $\Delta$^14^C of 2001 bulk soil incubations and corresponding bulk soil**

>*Caption:* $\Delta$^14^C of bulk soil and respired CO~2~ by depth for each site in 2001. Points show mean of three replicate profiles for bulk soils and mean of laboratory duplicates for respired CO~2~. The incubated soil samples are a composite made by homogenizing subsamples from each of the three replicate profile samples by depth. Error bars show one standard deviation for bulk soil and the measured values from laboratory duplicates of the incubated composite samples.

```{r prep-inc-by-bulk-14c-plot}
# first merge mean 14C data from 2019 samples with composite incubation data
nms.inc.blk <- c("pm", "eco", "lyr_bot", "Year")
sra.19.inc.blk2 <- left_join(sra.19.bulk %>% mutate(., Year = as.factor(Year)),
                             sra.2019.inc.sum.df,
                             by = nms.inc.blk,
                             suffix = c(".bulk", ".inc"))
# 2001
sra.01.inc.blk2 <- left_join(sra.01.sum %>% mutate(., Year = as.factor(Year)),
                             sra.01.inc,
                             by = nms.inc.blk,
                             suffix = c(".bulk", ".inc"))
sra.01.inc.blk2$PMeco <- paste0(sra.01.inc.blk2$pm, sra.01.inc.blk2$eco)
# add depth factor
sra.01.inc.blk2 <- unsplit(
  lapply(split(sra.01.inc.blk2, sra.01.inc.blk2$PMeco), function(x) {
  x$depth <- seq(1, nrow(x))
  return(x) 
  }), 
sra.01.inc.blk2$PMeco)
sra.01.inc.blk2 <- sra.01.inc.blk2[which(sra.01.inc.blk2$lyr_bot < 35), ]
sra.01.inc.blk2$depth <- factor(sra.01.inc.blk2$depth)

# regress bulk vs. inc
min.inc.blk.19 <- min(sra.19.inc.blk2$d14c_l.inc,
                      sra.19.inc.blk2$d14c_l.bulk) # exclude highly negative incubation sample from GRwf
max.inc.blk.19 <- max(sra.19.inc.blk2$d14c_l.inc,
                      sra.19.inc.blk2$d14c_l.bulk)

# What is the ideal grouping/expected relationship?
```

```{r plot-inc-by-bulk-14c}
## look at combinatorial dataset
# sra.all.df.fx <- function(ls, year) {
#   cbind(bind_rows(lapply(ls, function(df) df[ , c("PMeco", "lyr_bot", "d14c")])),
#         year = year)
# }
# sra.all.df <- inner_join(
#   rbind(sra.all.df.fx(sra.2001.ls, 2001),
#         sra.all.df.fx(sra.2019.ls, 2019)),
#   rbind(sra.all.df.fx(sra.2001.inc.ls, 2001),
#         sra.all.df.fx(sra.2019.inc.ls, 2019)),
#   by = c("PMeco", "lyr_bot", "year"),
#   suffix = c("_bulk", "_inc"))
# sra.all.df <- sra.all.df %>%
#   mutate(PM = substr(PMeco, 1, 2),
#          ECO = substr(PMeco, 3, 4))
# 
# sra.all.df %>%
#   filter(d14c_inc > -130) %>%
#   ggplot(., aes(d14c_bulk, d14c_inc, color = PM)) +
#   geom_vline(xintercept = 0) +
#   geom_hline(yintercept = 0) +
#   geom_abline(slope = 1, intercept = 0) +
#   geom_smooth(method = "lm", formula = y ~ x, aes(fill = PM)) +
#   geom_point() +
#   scale_color_manual(name = "Parent material",
#                      values = c("AN" = "blue",
#                                 "BS" = "red",
#                                 "GR" = "darkgray"),
#                      labels = c("AN" = "andesite",
#                                 "BS" = "basalt",
#                                 "GR" = "granite")) +
#     scale_fill_manual(name = "Parent material",
#                      values = c("AN" = "blue",
#                                 "BS" = "red",
#                                 "GR" = "darkgray"),
#                      labels = c("AN" = "andesite",
#                                 "BS" = "basalt",
#                                 "GR" = "granite")) +
#   coord_fixed(xlim = c(-130, 200), ylim = c(-130, 200)) +
#   xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
#   ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
#   
# summary(lm(d14c_inc ~ d14c_bulk * PM, sra.all.df[sra.all.df$d14c_inc > -130, ]))

# join all data as means and sds
sra.all.sum.df <- left_join(
  bind_rows(sra.2001.sum.ls2) %>%
    select(PMeco, lyr_bot, d14c_mean, d14c_sd) %>%
    mutate(Year = 2001) %>%
    bind_rows(., 
              bind_rows(lapply(sra.2019.ls, function(df) {
                df %>%
                  filter(lyr_bot < 31) %>%
                  select(PMeco, lyr_bot, d14c) %>%
                  group_by(PMeco, lyr_bot) %>%
                  summarize(across(d14c, list(mean = mean, sd = sd))) %>%
                  mutate(Year = 2019)
                }))
            ),
  bind_rows(lapply(sra.2001.inc.ls, function(df) {
              df %>%
                select(PMeco, lyr_bot, d14c) %>%
                group_by(PMeco, lyr_bot) %>%
                summarize(across(d14c, list(mean = mean, sd = sd))) %>%
                mutate(Year = 2001) 
              })) %>%
  bind_rows(., 
            bind_rows(lapply(sra.2019.inc.ls, function(df) {
              df %>%
                select(PMeco, lyr_bot, d14c) %>%
                group_by(PMeco, lyr_bot) %>%
                summarize(across(d14c, list(mean = mean, sd = sd))) %>%
                mutate(Year = 2019)
              }))
            ), 
  by = c("PMeco", "lyr_bot", "Year"),
  suffix = c(".bulk", ".inc")) %>%
  mutate(PM = substring(PMeco, 1, 2),
         eco = substring(PMeco, 3, 4))

# Trend for means
# NB Year only, depth only models do not show significant interactions
# PM only model
emtrends(lm(d14c_mean.inc ~ d14c_mean.bulk * PM, sra.all.sum.df[sra.all.sum.df$d14c_mean.inc > -200, ]), pairwise ~ PM, var = "d14c_mean.bulk")
# ECO only model
emtrends(lm(d14c_mean.inc ~ d14c_mean.bulk * eco, sra.all.sum.df[sra.all.sum.df$d14c_mean.inc > -200, ]), pairwise ~ eco, var = "d14c_mean.bulk")


# lapply(split(sra.all.sum.df, sra.all.sum.df$eco), function(df) {
#   summary(lm(d14c_mean.inc ~ d14c_mean.bulk * PM, df))
# })

# # Deming regression (accounts for error in x and y terms)
# sra.dem <- lapply(split(sra.all.sum.df, sra.all.sum.df$PM), function(df) {
#   deming(d14c_mean.inc ~ d14c_mean.bulk,
#        data = df, xstd = d14c_sd.inc, ystd = d14c_sd.bulk)
# })

# all depths and years together, by PM
fig.n <- fig.n + 1
sra.19.inc.blk2  %>%
  bind_rows(., sra.01.inc.blk2[ , which(names(sra.19.inc.blk2) %in% names(sra.01.inc.blk2))]) %>%
  mutate(depth = factor(lyr_bot),
         ecoYear = paste0(eco, " (", Year, ")")) %>%
  filter(d14c_mean.inc > -200) %>%
  ggplot(., aes(d14c_mean.bulk, d14c_mean.inc, color = pm)) +
  # geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_vline(xintercept = 0) +
  # geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_abline(slope = 1, intercept = 0) +
  geom_point(aes(color = pm, shape = ecoYear), size = 3) +
  geom_smooth(method = "lm", formula = y ~ x, se = FALSE) +
  geom_errorbarh(
    aes(xmin = d14c_l.bulk, 
        xmax = d14c_u.bulk,
        color = pm), 
    height = 1.5) +
  geom_errorbar(
    aes(ymin = d14c_l.inc, 
        ymax = d14c_u.inc,
        color = pm), 
    width = 1.5) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Ecosystem (year)",
                     values = c("warm (2019)" = 0,
                                "cool (2019)" = 2,
                                "cold (2019)" = 1,
                                "warm (2001)" = 15,
                                "cool (2001)" = 17,
                                "cold (2001)" = 16)) +
  coord_fixed(xlim = c(-100, 200), ylim = c(-100, 200)) +
  xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
  ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
  # facet_grid(rows = vars(depth)) +
  theme_bw() +
  theme(panel.grid = element_blank())

sra.19.inc.blk2  %>%
  bind_rows(., sra.01.inc.blk2[ , which(names(sra.19.inc.blk2) %in% names(sra.01.inc.blk2))]) %>%
  mutate(depth = factor(lyr_bot),
         ecoYear = paste0(eco, " (", Year, ")")) %>%
  filter(d14c_mean.inc > -200) %>%
  ggplot(., aes(d14c_mean.bulk, d14c_mean.inc, color = eco)) +
  # geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
  geom_vline(xintercept = 0) +
  # geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
  geom_hline(yintercept = 0) +
  geom_abline(slope = 1, intercept = 0) +
  geom_point(aes(color = eco, shape = ecoYear), size = 3) +
  geom_smooth(method = "lm", formula = y ~ x, se = FALSE) +
  geom_errorbarh(
    aes(xmin = d14c_l.bulk, 
        xmax = d14c_u.bulk,
        color = eco), 
    height = 1.5) +
  geom_errorbar(
    aes(ymin = d14c_l.inc, 
        ymax = d14c_u.inc,
        color = eco), 
    width = 1.5) +
  scale_color_manual(name = "Climate",
                     values = c("warm" = warm,
                                "cool" = cool,
                                "cold" = cold)) +
  scale_shape_manual(name = "Ecosystem (year)",
                     values = c("warm (2019)" = 0,
                                "cool (2019)" = 1,
                                "cold (2019)" = 2,
                                "warm (2001)" = 15,
                                "cool (2001)" = 16,
                                "cold (2001)" = 17)) +
  coord_fixed(xlim = c(-100, 200), ylim = c(-100, 200)) +
  xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
  ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
  # facet_grid(rows = vars(depth)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# # 2001
# sra.01.inc.blk2 %>%
#   filter(d14c_mean.bulk > -100 & d14c_mean.inc > -100) %>%
#   ggplot(., aes(d14c_mean.bulk, d14c_mean.inc, color = pm, shape = eco, group = pm)) +
#   # geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
#   geom_vline(xintercept = 0) +
#   # geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
#   geom_hline(yintercept = 0) +
#   geom_abline(slope = 1, intercept = 0) +
#   geom_point(size = 3) +
#   geom_smooth(method = "lm", formula = y ~ x, se = FALSE) +
#   geom_errorbarh(
#     aes(xmin = d14c_l.bulk, 
#         xmax = d14c_u.bulk,
#         color = pm), 
#     height = 1.5) +
#   geom_errorbar(
#     aes(ymin = d14c_l.inc, 
#         ymax = d14c_u.inc,
#         color = pm), 
#     width = 1.5) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = "blue",
#                                 "basalt" = "red",
#                                 "granite" = "darkgray")) +
#   scale_shape_manual(name = "Ecosystem",
#                      values = c("warm" = 15,
#                                 "cool" = 16,
#                                 "cold" = 17)) +
#   coord_fixed(xlim = c(-100, 200), ylim = c(-100, 200)) +
#   xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
#   ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
#   facet_grid(rows = vars(depth)) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
# 
# # 2019
# fig.n <- fig.n + 1
# sra.19.inc.blk2 %>%
#   mutate(depth = factor(lyr_bot)) %>%
#   ggplot(., aes(d14c_mean.bulk, d14c_mean.inc, color = pm, shape = eco, group = pm)) +
#   # geom_vline(xintercept = atm.d14.2019, linetype = "dashed") +
#   geom_vline(xintercept = 0) +
#   # geom_hline(yintercept = atm.d14.2019, linetype = "dashed") +
#   geom_hline(yintercept = 0) +
#   geom_abline(slope = 1, intercept = 0) +
#   geom_point(size = 3) +
#   geom_smooth(method = "lm", formula = y ~ x, se = FALSE) +
#   geom_errorbarh(
#     aes(xmin = d14c_l.bulk, 
#         xmax = d14c_u.bulk,
#         color = pm), 
#     height = 1.5) +
#   geom_errorbar(
#     aes(ymin = d14c_l.inc, 
#         ymax = d14c_u.inc,
#         color = pm), 
#     width = 1.5) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = "blue",
#                                 "basalt" = "red",
#                                 "granite" = "darkgray")) +
#   scale_shape_manual(name = "Ecosystem",
#                      values = c("warm" = 0,
#                                 "cool" = 1,
#                                 "cold" = 2)) +
#   coord_fixed(xlim = c(-100, 200), ylim = c(-100, 200)) +
#   xlab(expression('Bulk soil '*Delta*''^14*'C (‰)')) +
#   ylab(expression('Incubation '*Delta*''^14*'C (‰)')) +
#   facet_grid(rows = vars(depth)) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. Regression of 2019 bulk soil incubations and corresponding bulk soil $\Delta$^14^C**

>*Caption:* Regressions of $\Delta$^14^C of bulk soil and respired CO~2~ by depth for each site in 2019. Error bars show one standard deviation for bulk soil, points show mean of three replicate profiles for bulk soils and single observations for respired CO~2~.

## Time series: $\Delta$^14^C by depth (as measured)
```{r plot-timeseries-meas-depths}
# combine '01, '09, '19 data
sra.01.19.raw <- rbind(bind_rows(sra.2001.sum.ls),
                       bind_rows(sra.2019.sum.ls))
sra.2009.df <- sra.09.sum[ , which(names(sra.09.sum) %in% names(sra.01.19.raw))]
sra.2009.df <- cbind(sra.2009.df, 
                     fm = NA,
                     d14c_sd = NA,
                     fm_sd = NA,
                     d14c_u = NA,
                     d14c_l = NA)
sra.01.09.19.raw <- rbind(sra.01.19.raw, sra.2009.df)
sra.01.09.19.raw$Year <- factor(c(rep(2001, nrow(bind_rows(sra.2001.sum.ls))),
                                  rep(2019, nrow(bind_rows(sra.2019.sum.ls))),
                                  rep(2009, nrow(sra.2009.df))),
                                levels = c("2001", "2009", "2019"))

# plot
# w/ ribbons
# sra.01.09.19.raw %>%
#   mutate(PMeco_year = paste0(PMeco, Year),
#          eco = factor(ifelse(ECO == "pp", "warm",
#                       ifelse(ECO == "wf", "cool", "cold")),
#                       levels = c("warm", "cool", "cold")),
#          d14c_u = d14c + d14c_sd,
#          d14c_l = d14c - d14c_sd,
#          pm = ifelse(PM == "AN", "andesite",
#                      ifelse(PM == "BS", "basalt", "granite"))) %>%
#   ggplot(., aes(d14c, lyr_bot, group = PMeco_year)) +
#   geom_vline(xintercept = 0) +
#   geom_hline(yintercept = 0) +
#   geom_ribbon(aes(xmin = d14c_l, xmax = d14c_u, fill = pm, alpha = Year, group = PMeco_year),
#               color = NA, show.legend = FALSE) +
#   geom_point(aes(fill = pm, color = pm, shape = eco, alpha = Year), size = 2) +
#   geom_point(aes(shape = eco), color = "black", size = 3) +
#   geom_path(aes(linetype = Year, color = pm), size = 0.7) +
#   scale_y_reverse() +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = "blue", 
#                                 "basalt" = "red", 
#                                 "granite" = "darkgray")) +
#   scale_fill_manual(name = "Parent material",
#                     values = c("andesite" = "blue", 
#                                "basalt" = "red", 
#                                "granite" = "darkgray")) +
#   scale_shape_manual(name = "Ecosystem",
#                      values = c("warm" = 22, 
#                                 "cool" = 21, 
#                                 "cold" = 24)) +
#   scale_alpha_manual(values = c("2001" = .6,
#                                 "2009" = 0.4,
#                                 "2019" = 0.2)) +
#   facet_grid(rows = vars(eco), cols = vars(pm)) +
#   xlab(expression(Delta*''^14*'C (‰)')) +
#   ylab("Depth (cm)") +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())

# litter
sra.2019.inc.L.df <- data.frame(
  sra.2019.inc_L.df %>%
    group_by(Year, PM, eco, lyr_bot, PMeco) %>%
    summarize(across(.cols = d14c, 
                     .fns = list(mean = mean, min = min, max = max))) %>%
    rename(year = Year, d14c = d14c_mean) %>%
    mutate(eco = factor(ifelse(eco == "pp", "warm",
                      ifelse(eco == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
           pm = ifelse(PM == "AN", "andesite",
                       ifelse(PM == "BS", "basalt", "granite"))))
# for plotting below
sra.2019.inc.L.df2 <- sra.2019.inc.L.df %>%
  rename(d14c_l = d14c_min,
         d14c_u = d14c_max) %>%
  mutate(PMeco_year = paste0(PMeco, year))

# with error bars, all depths
sra.01.09.19.raw %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  ggplot(., aes(d14c, lyr_bot, group = PMeco_year)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_point(aes(fill = pm, color = pm, shape = eco, alpha = Year), size = 3.5) +
  geom_point(data = sra.2019.inc.L.df2, 
             aes(d14c, lyr_bot, color = pm, shape = eco), shape = 8, size = 3.5, show.legend = FALSE) +
  geom_path(aes(linetype = Year, color = pm), size = 0.7) +
  geom_errorbarh(
    aes(xmin = d14c_l,
        xmax = d14c_u,
        color = pm,
        alpha = Year),
    height = 1.5) +
  scale_y_reverse() +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue",
                                "basalt" = "red",
                                "granite" = "darkgray")) +
  scale_fill_manual(name = "Parent material",
                    values = c("andesite" = "blue",
                               "basalt" = "red",
                               "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15,
                                "cool" = 16,
                                "cold" = 17)) +
  scale_alpha_manual(values = c("2001" = 1,
                                "2009" = 0.6,
                                "2019" = 0.3)) +
  scale_linetype_manual(values = c("2001" = 1,
                                   "2009" = 2,
                                   "2019" = 3)) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# just topsoil, w/ error bars
fig.n <- fig.n + 1
sra.01.09.19.raw <- sra.01.09.19.raw[order(sra.01.09.19.raw$lyr_top), ]
sra.01.09.19.raw %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  ggplot(., aes(d14c, lyr_bot, group = PMeco_year)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_point(aes(fill = pm, color = pm, shape = eco, alpha = Year), size = 3) +
  geom_path(aes(linetype = Year, color = pm), size = 0.7) +
  geom_errorbarh(
    aes(xmin = d14c_l,
        xmax = d14c_u,
        color = pm,
        alpha = Year),
    height = 1.5) +
  scale_y_reverse(limits = c(41, 0)) +
  scale_x_continuous(limits = c(-160, 190)) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_fill_manual(name = "Parent material",
                    values = c("andesite" = "blue", 
                               "basalt" = "red", 
                               "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_alpha_manual(values = c("2001" = 1,
                                "2009" = 0.6,
                                "2019" = 0.3)) +
  scale_linetype_manual(values = c("2001" = "solid",
                                   "2009" = "dashed",
                                   "2019" = "dotted")) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. Time series of $\Delta$^14^C by depth, as measured**

>*Caption:* Points show mean of three profile replicates for 2001, 2009, and 2019 samples. Error bars show ± 1 standard deviation of the mean (only a single profile was analyzed in 2009). Stars show litter incubation $\Delta$^14^C-CO~2~ for 2019 samples as a point of reference.

## Spline fitting

Soils collected in both the 2001 and 2009 sampling campaigns were sampled by horizon, but the depth intervals differed between the two sampling years. In 2009, full profiles were excavated for each site, as opposed to the shorter profiles collected in 2001 from the GR and AN sites. Radiocarbon was measured on all three replicate profiles at each site for the 2001 samples, but only for one of the replicate profiles at each site in 2009, e.g. ANpp rep2, etc.

In order to compare the radiocarbon profiles between 2001, 2009, and 2019 we first interpolated both radiocarbon and carbon stock data at 1 cm intervals for each site in the datasets from each year. The carbon-stock-weighted radiocarbon values for any given target depth interval can then be calculated as a simple sum of the product of the carbon weight of each 1 cm increment (relative to the total carbon stock of the target depth interval) and its radiocarbon value. A monotonic cubic spline fit with Hyman filtering was used for the carbon stock interpolation (Wendt and Hauser 2013), and a mass-preserving spline was used to fit the radiocarbon data (Bishop, T.F.A., McBratney, A.B., Laslett, G.M., (1999) Modelling soil attribute depth functions with equal-area quadratic smoothing splines. Geoderma, 91(1-2): 27-45).

```{r cn-clean, include = FALSE}
elm_results_df <- bind_rows(unlist(elm_results_ls, recursive = FALSE))
# Split IDs
PMeco_rep_depth <- bind_rows(
  lapply(strsplit(elm_results_df$ID, "_"), function(x) { 
    df <-  data.frame(PMeco = x[2],
                      pro_rep = x[3],
                      depth = x[4])
    df$PM <- substr(df$PMeco, 1, 2)
    df$ECO <- substr(df$PMeco, 3, 4)
    return(df)
  })
)
elm_results_df <- cbind(elm_results_df, PMeco_rep_depth)
```

```{r soc-2001, include = FALSE}
# merge soc.2001.ls and sra.2001.ls to add SOC data
sra.2001.ls <- mapply(merge, sra.2001.ls, soc.2001.ls, SIMPLIFY = FALSE)

# calculate cumulative stocks
sra.2001.ls <- lapply(sra.2001.ls, function(df) {
  ls <- split(df, df$pro_name)
  ls <- lapply(ls, function(x) {
    x <- x[order(x$lyr_bot), ] # make sure to order data
    x$lyr_soc_cmtv <- NA
    for(i in seq_along(x$lyr_bot)) {
      if(i == 1) {
        x$lyr_soc_cmtv[i] <- x$lyr_soc_kgm2[i]
      } else {
        x$lyr_soc_cmtv[i] <- x$lyr_soc_kgm2[i] + x$lyr_soc_cmtv[i-1] 
      }
    }
    return(x)
  })
  return(unsplit(ls, df$pro_name))
})
save(sra.2001.ls, file = "sra.2001.ls.RData")
```

```{r spline-fm, include = FALSE}
### spline fit for fm
## bulk (split by pro rep)
# 2001
sra.2001.fm.sp <- lapply(sra.2001.ls, function(df) {
  lapply(split(df, df$pro_name), function(x) {
    depths(x) <- pro_name ~ lyr_top + lyr_bot
    x.mps <- mpspline(x, var.name = "fm")
    x.mps$var.1cm <- x.mps$var.1cm[1:max(x$lyr_bot)]
    return(x.mps)
  })
})
# 2009
sra.2009.fm.sp <- lapply(sra.2009.ls, function(x) {
  depths(x) <- pro_name ~ lyr_top + lyr_bot
  x.mps <- mpspline(x, var.name = "lyr_fraction_modern")
  x.mps$var.1cm <- x.mps$var.1cm[1:max(x$lyr_bot)]
  return(x.mps)
})
# 2019
sra.2019.fm.sp <- lapply(sra.2019.ls, function(df) {
  lapply(split(df, df$pro_name), function(x) {
    depths(x) <- pro_name ~ lyr_top + lyr_bot
    x.mps <- mpspline(x, var.name = "fm")
    x.mps$var.1cm <- x.mps$var.1cm[1:max(x$lyr_bot)]
    return(x.mps)
  })
})

## inc
# need min and max values for spline fits
# 2001
sra.2001.inc.fm.sp <- lapply(sra.2001.inc.sum.ls, function(df) {
    depths(df) <- PMeco ~ lyr_top + lyr_bot
    mean.mps <- mpspline(df, var.name = "fm_mean")
    min.mps <- mpspline(df, var.name = "fm_l")
    max.mps <- mpspline(df, var.name = "fm_u")
    return(list(mean.var.1cm = mean.mps$var.1cm[1:max(df$lyr_bot)],
                min.var.1cm = min.mps$var.1cm[1:max(df$lyr_bot)],
                max.var.1cm = max.mps$var.1cm[1:max(df$lyr_bot)]))
  })
# 2019
sra.2019.inc.fm.sp <- lapply(sra.2019.inc.sum.ls, function(df) {
  df <- df[-which(df$lyr_bot == 0), ]
  depths(df) <- PMeco ~ lyr_top + lyr_bot
  mean.mps <- mpspline(df, var.name = "fm_mean")
  min.mps <- mpspline(df, var.name = "fm_l")
  max.mps <- mpspline(df, var.name = "fm_u")
  return(list(mean.var.1cm = mean.mps$var.1cm[1:max(df$lyr_bot)],
              min.var.1cm = min.mps$var.1cm[1:max(df$lyr_bot)],
              max.var.1cm = max.mps$var.1cm[1:max(df$lyr_bot)]))
})
```

```{r spline-bd-soc-19, include = FALSE}
# Need SOC stock data for 2019 samples: use mass data from 2009 samples 
# spline fit for cmtv mass; specify 10cm depths
sra.2009.mass.sp <- lapply(sra.2009.ls, function(df) {
  depths(df) <- pro_name ~ lyr_top + lyr_bot
  df.mps <- mpspline(df, var.name = "mass_kgm2", d = seq(0, max(df$lyr_bot), by = 10))
  return(df.mps)
})

sra.2009.bd.sp <- lapply(sra.2009.ls, function(df) {
  depths(df) <- pro_name ~ lyr_top + lyr_bot
  df.mps <- mpspline(df, var.name = "BD_g_cm_3", d = seq(0, max(df$lyr_bot), by = 10))
  return(df.mps)
})

# calculate mean of 1cm mass predictions for each 2019 depth increment
mass_pred <- lapply(seq_along(sra.2009.mass.sp), function(i) {
  ls <- split(sra.2019.ls[[i]], sra.2019.ls[[i]]["pro_rep"]) # split each replicate profile
  ls <- lapply(ls, function(df) {
    t(sra.2009.mass.sp[[i]][["var.std"]])[1:nrow(df)] # mass_kgm2
    })
  return(unsplit(ls, sra.2019.ls[[i]]["pro_rep"]))
})

# calculate mean of 1cm BD predictions for each 2019 depth increment
bd_pred <- lapply(seq_along(sra.2009.bd.sp), function(i) {
  ls <- split(sra.2019.ls[[i]], sra.2019.ls[[i]]["pro_rep"]) # split each replicate profile
  ls <- lapply(ls, function(df) {
    t(sra.2009.bd.sp[[i]][["var.std"]])[1:nrow(df)] # mass_kgm2
    })
  return(unsplit(ls, sra.2019.ls[[i]]["pro_rep"]))
})

# merge predicted mass values with sra.2019.ls
nms <- names(sra.2019.ls)
sra.2019.ls <- lapply(seq_along(sra.2019.ls), function(df) {
  cbind(sra.2019.ls[[df]], bd_g_cm3 = bd_pred[[df]])
})
names(sra.2019.ls) <- nms

# add c conc
sra.2019.ls <- lapply(sra.2019.ls, function(df) {
  df$depth <- paste0(df$lyr_top, "-", df$lyr_bot)
  df <- merge(df, elm_results_df[ , c("PMeco", "pro_rep", "depth", "C", "N")], by = c("PMeco", "depth", "pro_rep"))
  return(df)
})

## calculate stocks and cumulative stocks
cstock.fx <- function(ls, mass, bd, C) {
  lapply(ls, function(df) {
    if (is.na(mass)) {
     df$lyr_soc <- df[[bd]] * df[[C]] * (df$lyr_bot - df$lyr_top) * 10^-1 
    } else {
     df$lyr_soc <- df[[mass]] * df[[C]] * 10^-2 
    }
    pro_ls <- split(df, df$pro_name)
    pro_ls <- lapply(pro_ls, function(x) {
      x$lyr_soc_cmtv <- NA
      for(i in seq_along(x$lyr_bot)) {
        if(i == 1) {
          x$lyr_soc_cmtv[i] <- x$lyr_soc[i]
        } else {
          x$lyr_soc_cmtv[i] <- x$lyr_soc[i] + x$lyr_soc_cmtv[i-1] 
        }
      }
      return(x)
    })
    return(unsplit(pro_ls, df$pro_name))
  })
}
# 2019
sra.2019.ls <- cstock.fx(sra.2019.ls, mass = NA, bd = "bd_g_cm3", "C")
# 2009
sra.2009.ls <- cstock.fx(sra.2009.ls, "mass_kgm2", bd = NA, "C_pct")

# save
save(sra.2019.ls, file = "sra.2019.ls.RData")
# write.csv(bind_rows(sra.2019.ls), file = "sra.2019.df.csv")

# make df
sra.2019.sum.df <- bind_rows(
  lapply(sra.2019.ls, function(df) {
    df %>%
      group_by(PMeco, lyr_top, lyr_bot) %>%
      summarize(across(c(fm, bd_g_cm3, C, N, lyr_soc), .fns = list(mean = mean, sd = sd))) %>%
      select(-bd_g_cm3_sd) %>%
      data.frame
  })
  , .id = "PMeco")
# write.csv(sra.2019.sum.df, file = "/Users/jeff/Desktop/sra.2019.sum.csv")
```

```{r spline-soc, include = FALSE}
# spline fit for carbon stocks (for calc weighted averages)
depth.spline <- function(x) {
  sp <- spline(x, method = "hyman") # fit monotonic cubic spline
  sp.ss <- smooth.spline(sp) # convert to class "spline" with smooth.spline fxn
  std <- seq(0, 100) # in cm (linear beyond last measured depth)
  sp <- predict(sp.ss, std) 
  df <- data.frame(sp)
  colnames(df) <- c("lyr_bot","lyr_soc") # where lyr_soc = cumulative SOC in output
  for(i in seq_along(df$lyr_bot)) {
    if(i == 1) {
      df$lyr_soc[i] <- df$lyr_soc[i]
    } else {
      df$lyr_soc[i] <- df$lyr_soc[i + 1] - df$lyr_soc[i]
    }
  }
  df <- df[-1,]
  return(df[-length(df$lyr_soc), ])
}

## add (0, 0) point for (lyr_bot, lyr_cmtv_stock)
# 2001
sra.2001.sp.ls <- lapply(sra.2001.ls, function(df) {
  ls <- lapply(split(df, df$pro_name), function(x) {
    t0 <- data.frame(matrix(nrow = 1, ncol = ncol(x)))
    xy <- c(which(names(x) == "lyr_bot"), which(names(x) == "lyr_soc_cmtv"))
    t0[ , xy] <- 0
    names(t0) <- names(x)
    t0$pro_name <- unique(x$pro_name)
    return(rbind(t0, x))
  })
  return(bind_rows(ls))
})
sra.2001.sp.ls.avg <- lapply(sra.2001.sum.ls2, function(df) {
  xy <- df[ , c("lyr_bot", "lyr_soc_cmtv")]
  t0 <- data.frame(lyr_bot = 0, lyr_soc_cmtv = 0)
  return(rbind(t0, xy))
})
# 2009
sra.2009.sp.ls <- lapply(sra.2009.ls, function(df) {
  t0 <- data.frame(matrix(nrow = 1, ncol = ncol(df)))
  xy <- c(which(names(df) == "lyr_bot"), which(names(df) == "lyr_soc_cmtv"))
  t0[ , xy] <- 0
  names(t0) <- names(df)
  new <- rbind(t0, df)
  return(new)
})
# 2019
sra.2019.sp.ls <- lapply(sra.2019.ls, function(df) {
  ls <- lapply(split(df, df$pro_name), function(x) {
    t0 <- data.frame(matrix(nrow = 1, ncol = ncol(x)))
    xy <- c(which(names(x) == "lyr_bot"), which(names(x) == "lyr_soc_cmtv"))
    t0[ , xy] <- 0
    names(t0) <- names(x)
    t0$pro_name <- unique(x$pro_name)
    return(rbind(t0, x))
  })
  return(bind_rows(ls))
})

## run spline
# 2001
sra.2001.oc.sp <- lapply(sra.2001.sp.ls, function(df) {
  lapply(split(df, df$pro_name), function(x) {
    depth.spline(x[, c("lyr_bot", "lyr_soc_cmtv")])
  })
})
sra.2001.oc.sp.avg <- lapply(sra.2001.sp.ls.avg, function(df) {
  depth.spline(df)
})
# 2009
sra.2009.oc.sp <- lapply(sra.2009.sp.ls, function(x) {
  depth.spline(x[, c("lyr_bot", "lyr_soc_cmtv")])
})
# 2019
sra.2019.oc.sp <- lapply(sra.2019.sp.ls, function(df) {
  lapply(split(df, df$pro_name), function(x) {
    depth.spline(x[, c("lyr_bot", "lyr_soc_cmtv")])
  })
})
sra.2019.oc.sp.avg <- lapply(sra.2019.oc.sp, function(ls) {
  bind_rows(ls) %>%
    group_by(lyr_bot) %>%
    summarize(lyr_soc = mean(lyr_soc)) %>%
    data.frame
})
```

```{r plot-soc-cmtv}
# 2001
sra.2001.soc.df <- bind_rows(lapply(seq_along(sra.2001.oc.sp.avg), function(i) {
  NM <- names(sra.2001.oc.sp.avg)[i]
  PM <- substr(NM, 1, 2)
  ECO <- substr(NM, 3, 4)
  df <- data.frame(PM = PM, ECO = ECO, lyr_soc_30 = sum(sra.2001.oc.sp.avg[[i]][1:30, "lyr_soc"]))
  df$ECO <- factor(df$ECO, levels = c("pp", "wf", "rf"))
  return(df)
}))

# 2009
sra.2009.soc.df <- bind_rows(lapply(seq_along(sra.2009.oc.sp), function(i) {
  NM <- names(sra.2009.oc.sp)[i]
  PM <- substr(NM, 1, 2)
  ECO <- substr(NM, 3, 4)
  df <- data.frame(PM = PM, ECO = ECO, lyr_soc_30 = sum(sra.2009.oc.sp[[i]][1:30, "lyr_soc"]))
  df$ECO <- factor(df$ECO, levels = c("pp", "wf", "rf"))
  return(df)
}))

# 2019
sra.2019.soc.df <- bind_rows(lapply(seq_along(sra.2019.oc.sp.avg), function(i) {
  NM <- names(sra.2019.oc.sp.avg)[i]
  PM <- substr(NM, 1, 2)
  ECO <- substr(NM, 3, 4)
  df <- data.frame(PM = PM, ECO = ECO, lyr_soc_30 = sum(sra.2019.oc.sp.avg[[i]][1:30, "lyr_soc"]))
  df$ECO <- factor(df$ECO, levels = c("pp", "wf", "rf"))
  return(df)
}))

ggplot(sra.2019.soc.df, aes(PM, lyr_soc_30, fill = PM)) +
  geom_col(position = "dodge") +
  scale_fill_manual(values = c("AN" = andesite, 
                               "BS" = basalt,
                               "GR" = granite)) +
  facet_grid(cols = vars(ECO)) +
  theme_bw() +
  theme(panel.grid = element_blank())

ggplot(sra.2009.soc.df, aes(PM, lyr_soc_30, fill = PM)) +
  geom_col(position = "dodge") +
  scale_fill_manual(values = c("AN" = andesite, 
                               "BS" = basalt,
                               "GR" = granite)) +
  facet_grid(cols = vars(ECO)) +
  theme_bw() +
  theme(panel.grid = element_blank())

ggplot(sra.2001.soc.df, aes(PM, lyr_soc_30, fill = PM)) +
  geom_col(position = "dodge") +
  scale_fill_manual(values = c("AN" = andesite, 
                               "BS" = basalt,
                               "GR" = granite)) +
  facet_grid(cols = vars(ECO)) +
  theme_bw() +
  theme(panel.grid = element_blank())

# all together
sra.01.09.19.soc.df <- cbind(rbind(sra.2001.soc.df,
                                   sra.2009.soc.df,
                                   sra.2019.soc.df), 
                             year = rep(c(2001, 2009, 2019), each = 9))
sra.01.09.19.soc.df %>%
  mutate(PMyear = paste0(PM, year)) %>%
  ggplot(., aes(PMyear, lyr_soc_30, fill = PM, alpha = year)) +
  geom_col(position = "dodge") +
  scale_fill_manual(values = c("AN" = andesite, 
                               "BS" = basalt,
                               "GR" = granite)) +
  facet_grid(cols = vars(ECO)) +
  theme_bw() +
  theme(panel.grid = element_blank())
```

```{r cwt-d14c-01, include = FALSE}
## calculate stock weights
## 2001 depths
# order '01 data
sra.2001.sum.ls <- lapply(sra.2001.sum.ls, function(df) df[order(df$lyr_bot), ])
# 2009
cwt.01.09 <- lapply(seq_along(sra.2009.oc.sp), function(i) {
  d <- sra.2001.sum.ls[[i]][["lyr_bot"]] # map onto '01 depths
  c <- vector(mode = "list", length = length(d))
  for(j in seq_along(d)) {
    if(j == 1) {
      c[[j]] <- sra.2009.oc.sp[[i]][1:d[j], "lyr_soc"]
    } else {
      c[[j]] <- sra.2009.oc.sp[[i]][(d[j-1]+1):d[j], "lyr_soc"] 
    }
  }
  return(unlist(lapply(c, function(x) x/sum(x)))) # return weights
})
names(cwt.01.09) <- names(sra.2009.oc.sp)
# 2019 (second list level from profile reps)
cwt.01.19 <- lapply(seq_along(sra.2019.oc.sp), function(i) {
  lapply(sra.2019.oc.sp[[i]], function(df) {
    d <- sra.2001.sum.ls[[i]][["lyr_bot"]] # map onto '01 depths
    c <- vector(mode = "list", length = length(d))
    for(j in seq_along(d)) {
      if(j == 1) {
        c[[j]] <- df[1:d[j], "lyr_soc"]
      } else {
        c[[j]] <- df[(d[j-1]+1):d[j], "lyr_soc"] 
      }
    }
    return(unlist(lapply(c, function(x) x/sum(x))))  # return weights
  })
})
names(cwt.01.19) <- names(sra.2019.oc.sp)
# average cwt '19 samples, '01 depths
cwt.01.19.avg <- lapply(seq_along(sra.2019.oc.sp.avg), function(i) {
    d <- sra.2019.sum.ls[[i]][["lyr_bot"]]
    c <- vector(mode = "list", length = length(d))
    for(j in seq_along(d)) {
      if(j == 1) {
        c[[j]] <- sra.2001.oc.sp.avg[[i]][1:d[j], "lyr_soc"]
      } else {
        c[[j]] <- sra.2001.oc.sp.avg[[i]][(d[j-1]+1):d[j], "lyr_soc"] 
      }
    }
    return(unlist(lapply(c, function(x) x/sum(x))))
})
names(cwt.01.19.avg) <- names(sra.2019.oc.sp)

## calculate fm_wts
## '01 depths
# 2019
fm.wt.01.19 <- lapply(seq_along(cwt.01.19), function(i) {
  lapply(seq_along(cwt.01.19[[i]]), function(j) {
    df <- data.frame(cwt = cwt.01.19[[i]][[j]])
    df$fm <- sra.2019.fm.sp[[i]][[j]][["var.1cm"]][1:length(cwt.01.19[[i]][[j]])]
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.01.19) <- names(cwt.01.19)

# 2009
fm.wt.01.09 <- lapply(seq_along(cwt.01.09), function(i) {
  df <- data.frame(cwt = cwt.01.09[[i]])
  df$fm <- sra.2009.fm.sp[[i]][["var.1cm"]][1:length(cwt.01.09[[i]])]
  df$fm_wt <- df$fm * df$cwt
  return(df)
})
names(fm.wt.01.09) <- names(cwt.01.09)


## calculate weighted average of fm for each interval
## '01 depths
# 2009
sra.01.09.ls <- lapply(seq_along(sra.2001.sum.ls), function(i) {
  d <- sra.2001.sum.ls[[i]][["lyr_bot"]]
  f <- vector(mode = "list", length = length(d))
  for(j in seq_along(d)) {
    if(j == 1) {
      f[[j]] <- sum(fm.wt.01.09[[i]][1:d[j], "fm_wt"])
    } else {
      f[[j]] <- sum(fm.wt.01.09[[i]][(d[j-1]+1):d[j], "fm_wt"])
    }
  }
  return(cbind(sra.2001.sum.ls[[i]], fm_09 = unlist(f)))
})
names(sra.01.09.ls) <- names(fm.wt.01.09)
# 2019
sra.01.19.ls <- fm.wt.01.19 # initialize list with fm wt structure
sra.01.19.ls <- lapply(seq_along(sra.2001.sum.ls), function(i) {
  sra.01.19.ls[[i]] <- lapply(seq_along(fm.wt.01.19[[i]]), function(x) {
    d <- sra.2001.sum.ls[[i]][["lyr_bot"]]
    f <- vector(mode = "list", length = length(d))
    for(j in seq_along(d)) {
      if(j == 1) {
        f[[j]] <- sum(fm.wt.01.19[[i]][[x]][1:d[j], "fm_wt"])
      } else {
        f[[j]] <- sum(fm.wt.01.19[[i]][[x]][(d[j-1]+1):d[j], "fm_wt"])
      }
    }
    return(unlist(f)) 
  })
  fm <- lapply(seq_along(sra.01.19.ls[[i]][[1]]), function(z) {
    data.frame(fm_19_mean = mean(sapply(sra.01.19.ls[[i]], "[", z), na.rm = TRUE),
               fm_19_sd = sd(sapply(sra.01.19.ls[[i]], "[", z), na.rm = TRUE))
    })
  fm <- bind_rows(fm)
  return(fm)
})
sra.01.09.19.ls <- lapply(seq_along(sra.01.09.ls), function(i) {
  data.frame(sra.01.09.ls[[i]], sra.01.19.ls[[i]])
})
names(sra.01.09.19.ls) <- names(sra.01.09.ls)

## create tidy combined '01, '09, '19 data frame
nms <- c("PM", "ECO", "PMeco", "lyr_top", "lyr_bot", "fm", "fm_sd")
sra.01.09.19.df <- bind_rows(sra.01.09.19.ls)
sra.01.09.19 <- sra.01.09.19.df[, nms]
sra.01.09.19 <- rbind(cbind(sra.01.09.19, Year = as.character(2001)),
                      data.frame(sra.01.09.19[, nms[1:5]],
                                 fm = sra.01.09.19.df$fm_09,
                                 fm_sd = NA,
                                 Year = as.character(2009)),
                      data.frame(sra.01.09.19[, nms[1:5]],
                                 fm = sra.01.09.19.df$fm_19_mean,
                                 fm_sd = sra.01.09.19.df$fm_19_sd,
                                 Year = as.character(2019)))

# calc d14c from fm
sra.01.09.19$d14c <- calc_14c(sra.01.09.19$fm, as.numeric(as.character(sra.01.09.19$Year)))
sra.01.09.19$d14c_sd <- abs(sra.01.09.19$d14c - calc_14c(sra.01.09.19$fm + sra.01.09.19$fm_sd, as.numeric(as.character(sra.01.09.19$Year))))
```

```{r cwt-d14c-19, include = FALSE}
## calculate stock weights
## 2019 depths
# 2001
cwt.19.01 <- lapply(seq_along(sra.2001.oc.sp), function(i) {
  lapply(sra.2001.oc.sp[[i]], function(df) {
    d <- sra.2019.sum.ls[[i]][["lyr_bot"]]
    c <- vector(mode = "list", length = length(d))
    for(j in seq_along(d)) {
      if(j == 1) {
        c[[j]] <- df[1:d[j], "lyr_soc"]
      } else {
        c[[j]] <- df[(d[j-1]+1):d[j], "lyr_soc"] 
      }
    }
    return(unlist(lapply(c, function(x) x/sum(x))))
  })
})
names(cwt.19.01) <- names(sra.2001.oc.sp)
# 2001 mean
cwt.19.01.avg <- lapply(seq_along(sra.2001.oc.sp.avg), function(i) {
    d <- sra.2019.sum.ls[[i]][["lyr_bot"]]
    c <- vector(mode = "list", length = length(d))
    for(j in seq_along(d)) {
      if(j == 1) {
        c[[j]] <- sra.2001.oc.sp.avg[[i]][1:d[j], "lyr_soc"]
      } else {
        c[[j]] <- sra.2001.oc.sp.avg[[i]][(d[j-1]+1):d[j], "lyr_soc"] 
      }
    }
    return(unlist(lapply(c, function(x) x/sum(x))))
})
names(cwt.19.01.avg) <- names(sra.2001.oc.sp)

# 2009
cwt.19.09 <- lapply(seq_along(sra.2009.oc.sp), function(i) {
  d <- sra.2019.sum.ls[[i]][["lyr_bot"]]
  c <- vector(mode = "list", length = length(d))
  for(j in seq_along(d)) {
    if(j == 1) {
      c[[j]] <- sra.2009.oc.sp[[i]][1:d[j], "lyr_soc"]
    } else {
      c[[j]] <- sra.2009.oc.sp[[i]][(d[j-1]+1):d[j], "lyr_soc"] 
    }
  }
  return(unlist(lapply(c, function(x) x/sum(x))))
})
names(cwt.19.09) <- names(sra.2009.oc.sp)

## calculate fm_wts
## '19 depths
## bulk
# 2001
fm.wt.19.01 <- lapply(seq_along(cwt.19.01), function(i) {
  lapply(seq_along(cwt.19.01[[i]]), function(j) {
    df <- data.frame(cwt = cwt.19.01[[i]][[j]])
    df$fm <- sra.2001.fm.sp[[i]][[j]][["var.1cm"]][1:length(cwt.19.01[[i]][[j]])]
    # linear extrapolation for filling 20-30cm fm data
    fm_1_30 <- df$fm[1:30] # 0-30cm fm
    if(length(which(is.na(fm_1_30))) > 0) {
     ix <- which(is.na(fm_1_30))
     ix.min <- min(ix) # first is.na(fm)
     m <- fm_1_30[ix.min-1]-fm_1_30[ix.min-2] # slope at last two measurement points
     for(i in ix.min:30) {
      fm_1_30[i] <- fm_1_30[i - 1] + m 
     }
     df$fm[1:30] <- fm_1_30 
    }
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
})
names(fm.wt.19.01) <- names(cwt.19.01)

# '01 inc
fm.wt.19.01.inc <- lapply(seq_along(cwt.19.01.avg), function(j) {
  lapply(sra.2001.inc.fm.sp[[j]], function(fm) {
    df <- data.frame(cwt = cwt.19.01.avg[[j]][1:30])
    # linear extrapolation for filling 20-30cm fm data
    if (length(fm) >= 30) {
      fm_1_30 <- fm[1:30] # 0-30cm fm
    } else {
      fm_1_30 <- rep(NA, 30)
      fm_1_30[1:length(fm)] <- fm
      # first is.na(fm)
      ix.min <- min(which(is.na(fm_1_30)))
      # slope at last two measurement points
      m <- fm_1_30[ix.min - 1] - fm_1_30[ix.min - 2]
      for(x in ix.min:30) {
        fm_1_30[x] <- fm_1_30[x - 1] + m
      }
    }
    df$fm[1:30] <- fm_1_30 
    df$fm_wt <- df$fm * df$cwt
    return(df)
  })
})
names(fm.wt.19.01.inc) <- names(cwt.19.01.avg)

# 2009
fm.wt.19.09 <- lapply(seq_along(cwt.19.09), function(i) {
  df <- data.frame(cwt = cwt.19.09[[i]])
  df$fm <- sra.2009.fm.sp[[i]][["var.1cm"]][1:length(cwt.19.09[[i]])]
  df$fm_wt <- df$fm * df$cwt
  return(df)
})
names(fm.wt.19.09) <- names(cwt.19.09)

## calculate weighted average of fm for each interval
## '19 depths
# 2001
# calculate weighted spline values for each profile rep
sra.19.01.rep.ls <- fm.wt.19.01
sra.19.01.rep.ls <- lapply(seq_along(sra.2019.sum.ls), function(i) {
  sra.19.01.rep.ls[[i]] <- lapply(seq_along(fm.wt.19.01[[i]]), function(x) {
    d <- sra.2019.sum.ls[[i]][["lyr_bot"]]
    f <- vector(mode = "list", length = length(d))
    for(j in seq_along(d)) {
      if(j == 1) {
        f[[j]] <- sum(fm.wt.19.01[[i]][[x]][1:d[j], "fm_wt"])
      } else {
        f[[j]] <- sum(fm.wt.19.01[[i]][[x]][(d[j-1]+1):d[j], "fm_wt"])
      }
    }
    return(unlist(f)) 
  })})
sra.19.01.rep.ls <- lapply(seq_along(sra.19.01.rep.ls), function(i) {
  names(sra.19.01.rep.ls[[i]]) <- names(cwt.19.01[[i]])
  return(sra.19.01.rep.ls[[i]])
})
names(sra.19.01.rep.ls) <- names(fm.wt.19.01)
save(sra.19.01.rep.ls, file = "sra.19.01.rep.ls.RData")

# average reps
sra.19.01.ls <- lapply(seq_along(sra.2019.sum.ls), function(i) {
  fm <- lapply(seq_along(sra.19.01.rep.ls[[i]][[1]]), function(z) {
    data.frame(fm_01_mean = mean(sapply(sra.19.01.rep.ls[[i]], "[", z), na.rm = TRUE),
               fm_01_sd = sd(sapply(sra.19.01.rep.ls[[i]], "[", z), na.rm = TRUE))
    })
  return(bind_rows(fm))
})
names(sra.19.01.ls) <- names(fm.wt.19.01)

## '01 inc
sra.19.01.inc.ls <- lapply(fm.wt.19.01.inc, function(ls) {
  lapply(ls, function(df) {
    d <- c(10, 20, 30)
    f <- vector(mode = "list", length = length(d))
    for(j in seq_along(d)) {
      if(j == 1) {
        f[[j]] <- sum(df[1:d[j], "fm_wt"])
      } else {
        f[[j]] <- sum(df[(d[j-1]+1):d[j], "fm_wt"])
      }
    }
    return(unlist(f)) 
  })})

# 2009
sra.19.09.ls <- lapply(seq_along(sra.2019.sum.ls), function(i) {
  d <- sra.2019.sum.ls[[i]][["lyr_bot"]]
  f <- vector(mode = "list", length = length(d))
  for(j in seq_along(d)) {
    if(j == 1) {
      f[[j]] <- sum(fm.wt.19.09[[i]][1:d[j], "fm_wt"])
    } else {
      f[[j]] <- sum(fm.wt.19.09[[i]][(d[j-1]+1):d[j], "fm_wt"])
    }
  }
  return(cbind(sra.2019.sum.ls[[i]], fm_09 = unlist(f)))
})
names(sra.19.09.ls) <- names(sra.19.01.ls)

# combine
sra.19.01.09.ls <- lapply(seq_along(sra.19.01.ls), function(i) {
  data.frame(sra.19.01.ls[[i]], sra.19.09.ls[[i]])
})
names(sra.19.01.09.ls) <- names(sra.19.01.ls)

## create tidy combined '01, '09, '19 data frame
sra.19.01.09.df <- bind_rows(sra.19.01.09.ls)
sra.19.01.09 <- sra.19.01.09.df[, nms]
sra.19.01.09 <- rbind(data.frame(sra.19.01.09[, nms[1:5]],
                                 fm = sra.19.01.09.df$fm_01_mean,
                                 fm_sd = sra.19.01.09.df$fm_01_sd,
                                 Year = as.character(2001)),
                     data.frame(sra.19.01.09[, nms[1:5]],
                                fm = sra.19.01.09.df$fm_09,
                                fm_sd = NA,
                                Year = as.character(2009)),
                     cbind(sra.19.01.09, Year = as.character(2019)))

# calc d14c from fm
sra.19.01.09$d14c <- calc_14c(sra.19.01.09$fm, as.numeric(as.character(sra.19.01.09$Year)))
sra.19.01.09$d14c_sd <- abs(sra.19.01.09$d14c - calc_14c(sra.19.01.09$fm + sra.19.01.09$fm_sd, as.numeric(as.character(sra.19.01.09$Year))))
save(sra.19.01.09, file = "sra.19.01.09.RData")
```

```{r cwt-19-01-0_30cm}
#### 0-30cm
### bulk
## 2019
# SOC weights
cwt.19_30 <- lapply(seq_along(sra.2019.oc.sp), function(i) {
  lapply(sra.2019.oc.sp[[i]], function(df) {
    d <- 30
    c <- df[1:d, "lyr_soc"]
    return(unlist(lapply(c, function(x) x/sum(c))))
  })
})
names(cwt.19_30) <- names(sra.2019.oc.sp)
# FM wts
fm.wt.19_30 <- lapply(seq_along(cwt.19_30), function(i) {
  lapply(seq_along(cwt.19_30[[i]]), function(j) {
    df <- data.frame(cwt = cwt.19_30[[i]][[j]])
    df$fm <- sra.2019.fm.sp[[i]][[j]][["var.1cm"]][1:length(cwt.19_30[[i]][[j]])]
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.19_30) <- names(cwt.19_30)
# summarize over 0-30cm
sra.19.rep.30.ls <- lapply(seq_along(1:9), function(i) {
  lapply(seq_along(fm.wt.19_30[[i]]), function(x) {
    d <- 30
    f <- sum(fm.wt.19_30[[i]][[x]][1:d, "fm_wt"])
    return(unlist(f)) 
  })
})
names(sra.19.rep.30.ls) <- names(fm.wt.19_30)
sra.19.avg.30.ls <- lapply(seq_along(sra.2019.sum.ls), function(i) {
  fm <- lapply(seq_along(sra.19.rep.30.ls[[i]][[1]]), function(z) {
    data.frame(fm_19_mean = mean(sapply(sra.19.rep.30.ls[[i]], "[", z), na.rm = TRUE),
               fm_19_sd = sd(sapply(sra.19.rep.30.ls[[i]], "[", z), na.rm = TRUE))
    })
  return(bind_rows(fm))
})
names(sra.19.avg.30.ls) <- names(fm.wt.19_30)

## 2001
# SOC weights
cwt.01_30 <- lapply(sra.2001.oc.sp, function(ls) {
  lapply(ls, function(df) {
    d <- 30
    c <- df[1:d, "lyr_soc"]
    return(unlist(lapply(c, function(x) x/sum(c))))
  })
})
names(cwt.01_30) <- names(sra.2001.oc.sp)
# FM wts
fm.wt.01_30 <- lapply(seq_along(cwt.01_30), function(i) {
  lapply(seq_along(cwt.01_30[[i]]), function(j) {
    df <- data.frame(cwt = cwt.01_30[[i]][[j]])
    df$fm <- sra.2001.fm.sp[[i]][[j]][["var.1cm"]][1:length(cwt.01_30[[i]][[j]])]
    # linear extrapolation for filling 20-30cm fm data
    fm_1_30 <- df$fm[1:30] # 0-30cm fm
    if(length(which(is.na(fm_1_30))) > 0) {
     ix <- which(is.na(fm_1_30))
     ix.min <- min(ix) # first is.na(fm)
     m <- fm_1_30[ix.min-1]-fm_1_30[ix.min-2] # slope at last two measurement points
     for(i in ix.min:30) {
      fm_1_30[i] <- fm_1_30[i - 1] + m 
     }
     df$fm[1:30] <- fm_1_30 
    }
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.01_30) <- names(cwt.01_30)
# summarize over 0-30cm
sra.01.rep.30.ls <- lapply(seq_along(1:9), function(i) {
  lapply(seq_along(fm.wt.01_30[[i]]), function(x) {
    d <- 30
    f <- sum(fm.wt.01_30[[i]][[x]][1:d, "fm_wt"])
    return(unlist(f)) 
  })
})
names(sra.01.rep.30.ls) <- names(fm.wt.01_30)
sra.01.avg.30.ls <- lapply(seq_along(1:9), function(i) {
  fm <- lapply(seq_along(sra.01.rep.30.ls[[i]][[1]]), function(z) {
    data.frame(fm_01_mean = mean(sapply(sra.01.rep.30.ls[[i]], "[", z), na.rm = TRUE),
               fm_01_sd = sd(sapply(sra.01.rep.30.ls[[i]], "[", z), na.rm = TRUE))
    })
  return(bind_rows(fm))
})
names(sra.01.avg.30.ls) <- names(fm.wt.01_30)

### inc
## 2019
# SOC weights (site average)
cwt.19_30.avg <- lapply(cwt.19_30, function(ls) {
  apply(bind_rows(ls), 1, mean)
})
# Flux weights (site average)

cwt.19_30.avg <- lapply(cwt.19_30, function(ls) {
  apply(bind_rows(ls), 1, mean)
})
# FM weights
fm.wt.19.30.inc <- lapply(seq_along(cwt.19_30.avg), function(j) {
  lapply(sra.2019.inc.fm.sp[[j]], function(fm) {
    df <- data.frame(cwt = cwt.19_30.avg[[j]])
    df$fm <- fm
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.19.30.inc) <- names(cwt.19_30.avg)
# summarize over 0-30cm
sra.19.30.inc.ls <- lapply(fm.wt.19.30.inc, function(ls) {
  ls <- lapply(ls, function(df) sum(df$fm_wt))
  names(ls) <- c("fm_19_mean", "fm_19_min", "fm_19_max")
  return(data.frame(bind_rows(ls)))
})
names(sra.19.30.inc.ls) <- names(cwt.19_30.avg)

## 2001
# SOC weights (site average)
cwt.01_30.avg <- lapply(cwt.01_30, function(ls) {
  apply(bind_rows(ls), 1, mean)
})
# FM weights
fm.wt.01.30.inc <- lapply(seq_along(cwt.01_30.avg), function(j) {
  lapply(sra.2001.inc.fm.sp[[j]], function(fm) {
    df <- data.frame(cwt = cwt.01_30.avg[[j]])
    df$fm <- fm[1:length(cwt.01_30.avg[[j]])]
    # linear extrapolation for filling 20-30cm fm data
    fm_1_30 <- df$fm[1:30] # 0-30cm fm
    if(length(which(is.na(fm_1_30))) > 0) {
     ix <- which(is.na(fm_1_30))
     ix.min <- min(ix) # first is.na(fm)
     m <- fm_1_30[ix.min-1]-fm_1_30[ix.min-2] # slope at last two measurement points
     for(i in ix.min:30) {
      fm_1_30[i] <- fm_1_30[i - 1] + m 
     }
     df$fm[1:30] <- fm_1_30 
    }
    df$fm_wt <- df$fm * df$cwt
  return(df)
  })
}) 
names(fm.wt.01.30.inc) <- names(cwt.01_30.avg)
# summarize over 0-30cm
sra.01.30.inc.ls <- lapply(fm.wt.01.30.inc, function(ls) {
  ls <- lapply(ls, function(df) sum(df$fm_wt))
  names(ls) <- c("fm_01_mean", "fm_01_min", "fm_01_max")
  return(data.frame(bind_rows(ls)))
})
names(sra.01.30.inc.ls) <- names(cwt.01_30.avg)

## df for linear modeling
# bulk
sra.blk.rep.30.ls <- lapply(seq_along(sra.01.rep.30.ls), function(i) {
  blk.01 <- data.frame(fm_blk = do.call(rbind, sra.01.rep.30.ls[[i]]),
                       year = 2001)
  blk.19 <- data.frame(fm_blk = do.call(rbind, sra.19.rep.30.ls[[i]]),
                       year = 2019)
  rbind(blk.01, blk.19) %>%
    mutate(d14c_blk = calc_14c(fm_blk, year))
})
names(sra.blk.rep.30.ls) <- names(sra.01.30.inc.ls)
sra.blk.rep.30.df <- bind_rows(sra.blk.rep.30.ls, .id = "PMeco")
# inc
inc.30.bind.fx <- function(ls, year_xx) {
  rbind(
    bind_rows(lapply(ls, "[", 2), .id = "PMeco") %>%
      rename(fm_inc = paste0("fm_", year_xx, "_min")),
    bind_rows(lapply(ls, "[", 3), .id = "PMeco") %>%
      rename(fm_inc = paste0("fm_", year_xx, "_max"))) %>%
    mutate(year = as.numeric(paste0("20", year_xx))) %>%
    mutate(d14c_inc = calc_14c(fm_inc, year))
}
sra.inc.rep.30.df <- rbind(inc.30.bind.fx(sra.01.30.inc.ls, "01"), 
                           inc.30.bind.fx(sra.19.30.inc.ls, "19"))
# combine
sra.blk.inc.rep.30.df <- merge(sra.blk.rep.30.df, sra.inc.rep.30.df, by = c("year", "PMeco"))
save(sra.blk.inc.rep.30.df, file = "sra.blk.inc.rep.30.df.RData")

## Combine mean data into a single data frame
# functions for converting fm to d14c and calculating sd
blk.14c.sd.fx <- function(df, year_xx) {
  date <- as.numeric(paste0(20, year_xx))
  df$fm_u <- df[[paste0("fm_", year_xx, "_mean")]] + df[[paste0("fm_", year_xx, "_sd")]]
  df$d14c_u <- calc_14c(df$fm_u, date)
  df[[paste0("d14c_", "mean")]] <- calc_14c(df[[paste0("fm_", year_xx, "_mean")]], date)
  df[[paste0("d14c_", "sd")]] <- df[[paste0("d14c_", "mean")]] - df$d14c_u
  df$year <- as.numeric(paste0(20, year_xx))
  return(df %>% select(c(starts_with("d14c"), year)) %>% select(-d14c_u))
}
inc.14c.sd.fx <- function(df, year_xx) {
  names(df) <- gsub(paste0("fm_", year_xx), "d14c", names(df))
  df_14c <- calc_14c(df, as.numeric(paste0(20, year_xx)))
  df_14c[[paste0("d14c_", "sd")]] <- sd(df_14c[ , 2:3])
  df_14c$year <- as.numeric(paste0(20, year_xx))
  return(df_14c[ , c(1, 4:5)])
}
# run functions and combine lists 
# 0-30cm data from '01 and '19
sra.30.blk.inc.ls <- lapply(
  list(lapply(sra.01.30.inc.ls, inc.14c.sd.fx, year_xx = "01"), 
       lapply(sra.19.30.inc.ls, inc.14c.sd.fx, year_xx = "19"),
       lapply(sra.01.avg.30.ls, blk.14c.sd.fx, year_xx = "01"),
       lapply(sra.19.avg.30.ls, blk.14c.sd.fx, year_xx = "19")),
  bind_rows, .id = "PMeco")
# reduce list to data frame, calculate difference of means and sd
sra.30.blk.inc.df <- rbind(merge(sra.30.blk.inc.ls[[1]],
                                 sra.30.blk.inc.ls[[3]],
                                 by = c("PMeco", "year"), suffixes = c("_inc", "_blk")),
                           merge(sra.30.blk.inc.ls[[2]],
                                 sra.30.blk.inc.ls[[4]],
                                 by = c("PMeco", "year"), suffixes = c("_inc", "_blk"))) %>%
  mutate(blk.inc = d14c_mean_blk - d14c_mean_inc,
         blk.inc.sd = sqrt(d14c_sd_blk^2/3 + d14c_sd_inc^2/2))
```

```{r plot-01-09-19-14c-profiles}
fig.n <- fig.n + 1
sra.01.09.19 %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  # filter(Year != "2009") %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, linetype = Year, group = PMeco_year)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_point(aes(alpha = Year), size = 3) +
  geom_path(aes(linetype = Year)) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  scale_y_reverse() +
  scale_x_continuous() +    
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_alpha_manual(values = c("2001" = 1,
                                "2009" = 0.6,
                                "2019" = 0.3)) +
  scale_linetype_manual(values = c("2001" = "solid",
                                 "2009" = "dashed",
                                 "2019" = "dotted")) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. Time series of bulk soil $\Delta$^14^C by 2001 depths (2001, 2009, 2019 samples)**

>*Caption:* Points for 2001 samples show the mean $\Delta$^14^C values at the measured depths. Points for 2009 and 2019 samples are spline-fitted estimates of $\Delta$^14^C predicted for the same depth intervals as measured in 2001. Error bars show ± 1 standard deviation of the mean of three replicate profiles for 2001 and 2019 samples (only a single profile was analyzed in 2009). 

```{r plot-19-01-09-14c-profiles}
fig.n <- fig.n + 1
sra.19.01.09 %>%
  mutate(PMeco_year = paste0(PMeco, Year),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  ggplot(., aes(d14c, lyr_bot, color = pm, shape = eco, linetype = Year, group = PMeco_year)) +
  geom_vline(xintercept = 0) +
  geom_hline(yintercept = 0) +
  geom_point(aes(alpha = Year), size = 3) +
  geom_path(aes(linetype = Year)) +
  geom_errorbarh(
    aes(xmin = d14c_l, 
        xmax = d14c_u,
        color = pm), 
    height = 1.5) +
  # scale_y_reverse(limits = c(30, 0)) +
  scale_y_reverse() +
  scale_x_continuous() +    
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_alpha_manual(values = c("2001" = 1,
                                "2009" = 0.6,
                                "2019" = 0.3)) +
  scale_linetype_manual(values = c("2001" = "solid",
                                   "2009" = "dashed",
                                   "2019" = "dotted")) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  xlab(expression(Delta*''^14*'C (‰)')) +
  ylab("Depth (cm)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```
>**Fig. `r {fig.n}`. Time series of bulk soil $\Delta$^14^C by depth (splined to 2019 depths)**

>*Caption:* Points for 2019 samples show the mean $\Delta$^14^C values at the measured depths. Points for 2001 and 2009 samples are spline-fitted estimates of $\Delta$^14^C predicted for the same depth intervals as measured in 2019. Error bars show ± 1 standard deviation of the mean of three replicate profiles for 2001 and 2019 samples (only a single profile was analyzed in 2009). 
>*NB: Only two depth intervals were measured at the cool and cold andesite sites (max depth of 27 and 28 cm, respectively), so linear extrapolation (using the slope of the last 1cm spline-fitted depth increment) was used to extend the profiles to 30 cm.*

```{r plot-by-depth-14C-timeseries}
# plot individual depths
fig.n <- fig.n + 1

# Atm
atm.14c <- data.frame(year = Datm[Datm$Date > 2000, "Date"],
                      d14c = Datm[Datm$Date > 2000, "NHc14"])
save(atm.14c, file = "atm.14c.RData")

# bulk 14C over time for 0-10, 10-20, 20-30 w/ atm
sra.19.01.09 %>%
  filter(lyr_bot < 31) %>%
  mutate(PMeco_depth = paste0(PMeco, lyr_bot),
         depth = factor(lyr_bot),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite")),
         year = as.numeric(as.character(Year))) %>%
  ggplot(., aes(year, d14c)) +
  geom_path(data = atm.14c) +
  geom_point(aes(color = pm, shape = eco), size = 3) +
  geom_path(aes(color = pm, group = PMeco_depth, linetype = depth), alpha = 0.3) +
  geom_errorbar(
    aes(ymin = d14c_l, 
        ymax = d14c_u,
        color = pm), 
    width = .5) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = "blue", 
                                "basalt" = "red", 
                                "granite" = "darkgray")) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_linetype_manual(name = "Depth (cm)",
                        labels = c("10" = "0-10",
                                   "20" = "10-20",
                                   "30" = "20-30"),
                        values = c("10" = 1,
                                   "20" = 2,
                                   "30" = 3)) +
  facet_grid(rows = vars(eco), cols = vars(pm)) +
  ylab(expression(Delta*''^14*'C (‰)')) +
  xlab("Year") +
  theme_bw() +
  theme(panel.grid = element_blank())

### incubation
## 2019
sra.2019.inc.df <- bind_rows(lapply(sra.2019.inc.ls, function(df) {
  data.frame(df %>%
               group_by(Year, PM, ECO, lyr_bot, PMeco) %>%
               summarize(
                 across(.cols = d14c, 
                        .fns = list(mean = mean, min = min, max = max))) %>%
               rename(year = Year, d14c = d14c_mean))
}))
save(sra.2019.inc.df, file = "sra.2019.inc.df.RData")
## 2001
sra.19.01.inc.df <- bind_rows(lapply(seq_along(sra.19.01.inc.ls), function(i) {
  PMeco <- names(sra.19.01.inc.ls)[i]
  d14c.ls <- lapply(sra.19.01.inc.ls[[i]], calc_14c, obs_date_y = 2001)
  df <- data.frame(d14c = d14c.ls[[1]],
                   d14c_min = d14c.ls[[2]],
                   d14c_max = d14c.ls[[3]],
                   lyr_bot = c(10, 20, 30),
                   PMeco = PMeco,
                   PM = substr(PMeco, 1, 2),
                   ECO = substr(PMeco, 3, 4),
                   year = 2001)
  return(df)
}))
# join
sra.19.01.inc <- rbind(sra.19.01.inc.df, sra.2019.inc.df)

# plot
sra.19.01.inc %>%
  mutate(PMeco_depth = paste0(PMeco, lyr_bot),
         depth = factor(lyr_bot),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite"))) %>%
  ggplot(., aes(year, d14c)) +
  geom_path(data = atm.14c) +
  geom_point(aes(color = eco, shape = eco), size = 3) +
  geom_point(data = sra.2019.inc.L.df, aes(color = eco), shape = 8, size = 3, show.legend = FALSE) +
  geom_path(aes(color = eco, group = PMeco), alpha = 0.3) +
  geom_errorbar(
    aes(ymin = d14c_min, 
        ymax = d14c_max,
        color = eco), 
    width = .5) +
  geom_errorbar(
    data = sra.2019.inc.L.df,
    aes(ymin = d14c_min, 
        ymax = d14c_max,
        color = eco), 
    width = .5) +
  scale_shape_manual(name = "Ecosystem",
                     values = c("warm" = 15, 
                                "cool" = 16, 
                                "cold" = 17)) +
  scale_y_continuous(limits = c(-40, 170)) +
  facet_grid(rows = vars(pm), cols = vars(depth)) +
  ylab(expression(Delta*''^14*'C (‰)')) +
  xlab("Year") +
  theme_bw() +
  theme(panel.grid = element_blank())

# plot inc and bulk together, by depth
sra.ts.all <- sra.19.01.09 %>%
  filter(lyr_bot < 31) %>%
  select(Year, PM, ECO, PMeco, lyr_bot, d14c, d14c_sd) %>%
  mutate(Type = "bulk",
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         year = as.numeric(as.character(Year))) %>%
  select(-d14c_sd, -Year) %>%
  bind_rows(.,
            sra.19.01.inc %>%
              select(year, PM, ECO, PMeco, lyr_bot, d14c, d14c_min, d14c_max) %>%
              rename(d14c_l = d14c_min,
                     d14c_u = d14c_max) %>%
              mutate(Type = "inc")
  ) %>%
  mutate(depth = factor(lyr_bot),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite")),
         ecoType = paste0(eco, " (", Type, ")"))

# Plot by depth
plot.ts.fx <- function(df) {
  df %>%
    filter(d14c > -200) %>%
    filter(year != 2009) %>%
    ggplot(., aes(year, d14c)) +
    geom_path(data = atm.14c) +
    geom_point(aes(color = pm, shape = ecoType), size = 3) +
    geom_path(aes(color = pm, linetype = Type), alpha = 0.3) +
    geom_errorbar(
      aes(ymin = d14c_l, 
          ymax = d14c_u,
          color = pm), 
      width = .5) +
    scale_color_manual(name = "Parent material",
                       values = c("andesite" = "blue", 
                                  "basalt" = "red", 
                                  "granite" = "darkgray")) +
    scale_shape_manual(name = "Ecosystem (type)",
                       values = c("warm (inc)" = 0,
                                  "cool (inc)" = 1,
                                  "cold (inc)" = 2,
                                  "warm (bulk)" = 15,
                                  "cool (bulk)" = 16,
                                  "cold (bulk)" = 17)) +
    facet_grid(rows = vars(eco), cols = vars(pm)) +
    ylab(expression(Delta*''^14*'C (‰)')) +
    xlab("Year") +
    theme_bw() +
    theme(panel.grid = element_blank())
}

# plots
lapply(split(sra.ts.all, sra.ts.all$depth), plot.ts.fx)

# # to save
# for(i in 1:3) ggsave(paste0(i, ".pdf"), lapply(split(sra.ts.all, sra.ts.all$depth), plot.ts.fx)[[i]])
```
>**Fig. `r {fig.n}`. Change in $\Delta$^14^C of bulk soil (panel a) and respired CO~2~ (panel b) over time relative to the atmosphere**

>*Caption:* Points for 2019 samples show the mean $\Delta$^14^C values at the measured depths. Points for 2001 and 2009 (bulk only) samples are spline-fitted estimates of $\Delta$^14^C predicted for the same depth intervals as measured in 2019. Error bars for bulk samples in panel (a) show ± 1 standard deviation of the mean of three replicate profiles for 2001 and 2019 samples (only a single profile was analyzed in 2009); error bars for incubation samples in panel (b) show the values of the two reps, while the point represents the mean. 
>*NB: Only two depth intervals were measured at the cool and cold andesite sites (max depth of 27 and 28 cm, respectively), so linear extrapolation (using the slope of the last 1cm spline-fitted depth increment) was used to extend the profiles to 30 cm.*

```{r min-data-ras18}
# load data
ras18.frc <- read_excel("/Users/jeff/sra-ts/data/external/sra_ras_sum/sierra_data_summary_2020.xlsx",
                        sheet = "2009_fraction_data")

# select only min cols and pivot longer
ras18_2 <- ras18.sum %>%
  select(`Fed (g/kg)`, `Feo (g/kg)`, `Alo (g/kg)`, `Alp (g/kg)`, `top mineral`, `bottom mineral`, pro_name) %>% 
  rename(lyr_top = `top mineral`,
         lyr_bot = `bottom mineral`) %>%
  pivot_longer(cols = c(`Fed (g/kg)`, 
                        `Feo (g/kg)`, 
                        `Alo (g/kg)`, 
                        `Alp (g/kg)`), 
               names_to = "mins", values_to = "conc") %>%
  data.frame()

# Calculate min stocks
ras18_3 <- ras18.sum %>%
  select(`Fed (g/kg)`, `Feo (g/kg)`, `Alo (g/kg)`, `Alp (g/kg)`, `top mineral`, `bottom mineral`, pro_name, BD_g_cm_3, Soil_finefraction, Thickness_cm) %>% 
  rename(lyr_top = `top mineral`,
         lyr_bot = `bottom mineral`) %>%
  pivot_longer(cols = c(`Fed (g/kg)`, 
                        `Feo (g/kg)`, 
                        `Alo (g/kg)`, 
                        `Alp (g/kg)`), 
               names_to = "mins", values_to = "conc") %>%
  mutate(mass = Thickness_cm * BD_g_cm_3 * Soil_finefraction * 10,
         min_stock = conc * mass * 10^-2) %>%
  data.frame()
ras18_3.ls <- lapply(split(ras18_3, ras18_3$mins), function(df) {
  lapply(split(df, df$pro_name), function(x) {
    x <- x[order(x$lyr_bot), ]
    # calc cmtv min stock
    x$min_stock_cmtv <- NA
    for(i in seq_along(x$lyr_bot)) {
      if(i == 1) {
        x$min_stock_cmtv[i] <- x$min_stock[i]
      } else {
        x$min_stock_cmtv[i] <- x$min_stock[i] + x$min_stock_cmtv[i-1] 
      }
    }
    x$mass_cmtv <- NA
    for(i in seq_along(x$lyr_bot)) {
      if(i == 1) {
        x$mass_cmtv[i] <- x$mass[i]
      } else {
        x$mass_cmtv[i] <- x$mass[i] + x$mass_cmtv[i-1] 
      }
    }
    return(x)
  })
})
ras18_3.sp.df <- bind_rows(lapply(ras18_3.ls, function(ls) {
  bind_rows(lapply(ls, function(x) {
    depths(x) <- pro_name ~ lyr_top + lyr_bot
    x.mps <- mpspline(x, var.name = "min_stock_cmtv")
    return(x.mps$var.1cm[30])
  }), .id = "pro_name") %>%
    pivot_longer(cols = everything(), names_to = "pro_name", values_to = "min_stock_cmtv")
}), .id = "min") %>%
  mutate(min = ifelse(min == "Alo (g/kg)", "Al_ox",
                      ifelse(min == "Alp (g/kg)", "Al_py",
                             ifelse(min == "Fed (g/kg)", "Fe_dc", "Fe_ox"))),
         PMeco = substr(pro_name, 1, 4)) %>%
  select(-pro_name)

# mass-weighted concentration
ras18_4.sp.ls <- lapply(ras18_3.ls, function(ls) {
  conc <- unlist(lapply(ls, function(x) {
    depths(x) <- pro_name ~ lyr_top + lyr_bot
    x.mps <- mpspline(x, var.name = "conc")
    df <- data.frame(conc = x.mps$var.1cm[1:30],
                     lyr_bot = seq(1, 30),
                     pro_name = substr(x.mps$idcol, 1, 4))
    return(split(df, df$pro_name))
    }), recursive = FALSE)
  mass <- unlist(lapply(seq_along(ls), function(i) {
    x <- ls[[i]][ , c("lyr_bot", "mass_cmtv")]
    t0 <- data.frame(matrix(nrow = 1, ncol = ncol(x)))
    names(t0) <- names(x)
    t0 <- 0
    x <- rbind(t0, x)
    sp <- spline(x, method = "hyman") # fit monotonic cubic spline
    sp.ss <- smooth.spline(sp) # convert to class "spline" with smooth.spline fxn
    std <- seq(0, 30) # depth in cm
    sp <- predict(sp.ss, std) 
    df <- data.frame(sp)
    colnames(df) <- c("lyr_bot", "mass_cmtv")
    df$pro_name <- substr(names(ls)[i], 1, 4)
    df <- df[-1, ]
    return(split(df, df$pro_name))
  }), recursive = FALSE)
  return(mapply(merge,
                mass,
                conc,
                SIMPLIFY = FALSE))
})

# calculate mass-weighted conc for 0-30cm
ras18_4.sp.df <- bind_rows(lapply(ras18_4.sp.ls, function(ls) {
  bind_rows(lapply(ls, function(df) {
    df <- df[order(df$lyr_bot), ]
    df$mass <- NA
    for (i in seq_along(df$mass)) {
      if (i == 1) {
        df$mass[i] <- df$mass_cmtv[i]
      } else {
        df$mass[i] <- df$mass_cmtv[i] - df$mass_cmtv[i-1]
      }
    }
    df$mass_wt <- df$mass/sum(df$mass)
    df$conc_30_wtd <- df$mass_wt * df$conc
    return(sum(df$conc_30_wtd))
  }), .id = "pro_name")
}), .id = "min") %>%
  pivot_longer(!min, names_to = "PMeco", values_to = "conc") %>%
  mutate(min = ifelse(min == "Alo (g/kg)", "Al_ox", 
                      ifelse(min == "Alp (g/kg)", "Al_py",
                             ifelse(min == "Fed (g/kg)", "Fe_dc", "Fe_ox"))))

# merge w/ 14C data
sra.all.30.min.conc.wtd <- merge(sra.30.blk.inc.df, ras18_4.sp.df, by = "PMeco") %>%
  mutate(pm = ifelse(substr(PMeco, 1, 2) == "AN", "andesite", 
                     ifelse(substr(PMeco, 1, 2) == "BS", "basalt", "granite")),
         eco = ifelse(substr(PMeco, 3, 4) == "pp", "warm", 
                      ifelse(substr(PMeco, 3, 4) == "wf", "cool", "cold")))
save(sra.all.30.min.conc.wtd, file = "sra.all.30.min.conc.wtd.RData")

# spline fits
# (should be mass-weighted...)
# also calculate for 0-30cm
ras18.split <- split(ras18_2, ras18_2$mins)
ras18.sp <- lapply(ras18.split, function(df) {
  ls <- lapply(split(df, df$pro_name), function(x) {
    depths(x) <- pro_name ~ lyr_top + lyr_bot
    x.mps <- mpspline(x, var.name = "conc", d = t(seq(0, 100, 10)))
    return(x.mps$var.std)
  })
  names(ls) <- unique(df$pro_name)
  return(ls)
})
names(ras18.sp) <- c("Al_ox", "Al_py", "Fe_dc", "Fe_ox")
ras18.sp.df <- data.frame(reduce(lapply(seq_along(ras18.sp), function(i) {
    df <- data.frame(t(bind_rows(ras18.sp[[i]])))
    names(df) <- unique(ras18.sum$pro_name)
    df$depth <- rownames(df)
    return(df %>%
             pivot_longer(!depth, names_to = "pro_name", values_to = names(ras18.sp)[i]))
  }),
  left_join,
  by = c("depth", "pro_name")
))
ras18.sp.df <- ras18.sp.df[-which(ras18.sp.df$depth == "soil depth"), ]
ras18.sp.df$lyr_bot <- rep(seq(10, 100, 10), each = 9)
ras18.sp.df <- ras18.sp.df[complete.cases(ras18.sp.df), ]
ras18.sp.df$PM <- substr(ras18.sp.df$pro_name, 1, 2)
ras18.sp.df$ECO <- substr(ras18.sp.df$pro_name, 3, 4)
save(ras18.sp.df, file = "ras18.sp.df.RData")

# reshape sra.ts.all w/ bulk and inc in separate cols
nms.inc.blk2 <- nms.inc.blk
nms.inc.blk2[[4]] <- "year"
sra.ts.all.blk.inc <- merge(sra.ts.all[sra.ts.all$Type == "bulk", ],
                            sra.ts.all[sra.ts.all$Type == "inc", c(nms.inc.blk2, "d14c", "d14c_u", "d14c_l")],
                            by = nms.inc.blk2,
                            suffixes = c("_bulk", "_inc")) %>%
  filter(year != 2009) %>%
  mutate(blk.inc = d14c_bulk - d14c_inc,
         blk.inc.sd = sqrt((d14c_u_bulk - d14c_bulk)^2 + apply(cbind(d14c_u_inc, d14c_l_inc), 1, var)))

# join w/ d14c
sra.all.min <- ras18.sp.df %>%
  mutate(pm = ifelse(PM == "AN", "andesite", ifelse(PM == "BS", "basalt", "granite")),
         eco = ifelse(ECO == "pp", "warm", ifelse(ECO == "wf", "cool", "cold"))) %>%
  # mutate(Al_nonCrys = Al_ox - Al_py,
  #        Fe_Crys = Fe_dc - Fe_ox) %>%
  select(-PM, -ECO, -pro_name) %>%
  left_join(sra.ts.all.blk.inc[ , c("pm", "eco", "lyr_bot", "year", "d14c_bulk", "d14c_u_bulk", "d14c_inc", "d14c_u_inc", "d14c_l_inc", "d14c_l_bulk", "blk.inc", "blk.inc.sd")], 
            ., 
            by = c("pm", "eco", "lyr_bot")) %>%
  pivot_longer(cols = c("Al_py", "Al_ox", "Fe_ox", "Fe_dc", 
                        # "Al_nonCrys", "Fe_Crys"
                        ), names_to = "min", values_to = "conc")

# Create min/14c df w/ 0-30cm 14C data
sra.all.30.min <- merge(sra.30.blk.inc.df, ras18_3.sp.df, by = "PMeco") %>%
  mutate(pm = ifelse(substr(PMeco, 1, 2) == "AN", "andesite", 
                     ifelse(substr(PMeco, 1, 2) == "BS", "basalt", "granite")),
         eco = ifelse(substr(PMeco, 3, 4) == "pp", "warm", 
                      ifelse(substr(PMeco, 3, 4) == "wf", "cool", "cold")))


# save
save(sra.all.min, file = "sra.all.min.RData")
save(sra.all.30.min, file = "sra.all.30.min.RData")
```

```{r plot-min-14c}
# bulk
sra.all.min %>%
  mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
         ecoYear = paste0(eco, " (", year, ")"),
         width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
  filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  ggplot(., aes(conc, d14c_bulk)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = d14c_l_bulk,
        ymax = d14c_u_bulk,
        color = pm,
        width = width)) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# inc
sra.all.min %>%
  mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
         ecoYear = paste0(eco, " (", year, ")"),
         width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
  filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  filter(lyr_bot == 30) %>%
  ggplot(., aes(conc, d14c_inc)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = d14c_l_inc,
        ymax = d14c_u_inc,
        color = pm,
        width = width)) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# bulk-inc
sra.all.min %>%
  mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
         ecoYear = paste0(eco, " (", year, ")"),
         width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
  filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  ggplot(., aes(conc, blk.inc)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = blk.inc - blk.inc.sd,
        ymax = blk.inc + blk.inc.sd,
        color = pm,
        width = width)) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# Fe_dc alone
#####
# warm
sra.all.min %>%
  filter(eco == "warm" & min == "Fe_dc") %>%
  mutate(ecoYear = paste0(eco, " (", year, ")")) %>%
  ggplot(., aes(conc, blk.inc)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = blk.inc - blk.inc.sd,
        ymax = blk.inc + blk.inc.sd,
        color = pm),
        width = 1.5) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15,
                                "warm (2019)" = 0)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# cool + cold
sra.all.min %>%
  filter(eco != "warm" & min == "Fe_dc") %>%
  mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
         ecoYear = paste0(eco, " (", year, ")")) %>%
  filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  ggplot(., aes(conc, blk.inc)) +
  geom_point(aes(color = pm, shape = ecoYear, size = depth)) +
  geom_errorbar(
    aes(ymin = blk.inc - blk.inc.sd,
        ymax = blk.inc + blk.inc.sd,
        color = pm),
        width = .5) +
  scale_color_manual(name = "Parent material",
                     values = c("andesite" = andesite,
                                "basalt" = basalt,
                                "granite" = granite)) +
  scale_shape_manual(name = "Climate (year)",
                     values = c("cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  scale_size_manual(name = "Depth",
                    values = (3:5)) +
  facet_wrap(vars(min), scales = "free") +
  ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
  xlab(expression('Concentration (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
#####

## by depth
#####
# # 10 cm
# sra.all.min %>%
#   mutate(ecoYear = paste0(eco, " (", year, ")"),
#          width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
#   filter(lyr_bot == 10) %>%
#   ggplot(., aes(conc, blk.inc)) +
#   geom_point(aes(color = pm, shape = ecoYear), size = 3) +
#   geom_errorbar(
#     aes(ymin = blk.inc - blk.inc.sd,
#         ymax = blk.inc + blk.inc.sd,
#         color = pm,
#         width = width)) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = andesite,
#                                 "basalt" = basalt,
#                                 "granite" = granite)) +
#   scale_shape_manual(name = "Climate (year)",
#                      values = c("warm (2001)" = 15, 
#                                 "cool (2001)" = 16, 
#                                 "cold (2001)" = 17,
#                                 "warm (2019)" = 0, 
#                                 "cool (2019)" = 1, 
#                                 "cold (2019)" = 2)) +
#   facet_wrap(vars(min), scales = "free") +
#   ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
#   xlab(expression('Concentration (g kg'^-1*')')) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
# 
# # 20 cm
# sra.all.min %>%
#   mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
#          ecoYear = paste0(eco, " (", year, ")"),
#          width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
#   filter(pmEcoDepth != "granitecold20") %>%
#   filter(lyr_bot == 20) %>%
#   ggplot(., aes(conc, blk.inc)) +
#   geom_point(aes(color = pm, shape = ecoYear), size = 3) +
#   geom_errorbar(
#     aes(ymin = blk.inc - blk.inc.sd,
#         ymax = blk.inc + blk.inc.sd,
#         color = pm,
#         width = width)) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = andesite,
#                                 "basalt" = basalt,
#                                 "granite" = granite)) +
#   scale_shape_manual(name = "Climate (year)",
#                      values = c("warm (2001)" = 15, 
#                                 "cool (2001)" = 16, 
#                                 "cold (2001)" = 17,
#                                 "warm (2019)" = 0, 
#                                 "cool (2019)" = 1, 
#                                 "cold (2019)" = 2)) +
#   facet_wrap(vars(min), scales = "free") +
#   ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
#   xlab(expression('Concentration (g kg'^-1*')')) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
# 
# # 30 cm
# sra.all.min %>%
#   mutate(pmEcoDepth = paste0(pm, eco, lyr_bot),
#          ecoYear = paste0(eco, " (", year, ")"),
#          width = ifelse(min == "Al_py" | min == "Fe_ox", .3, 1.5)) %>%
#   filter(pmEcoDepth != "granitecold30") %>%
#   filter(lyr_bot == 30) %>%
#   ggplot(., aes(conc, blk.inc)) +
#   geom_point(aes(color = pm, shape = ecoYear), size = 3) +
#   geom_errorbar(
#     aes(ymin = blk.inc - blk.inc.sd,
#         ymax = blk.inc + blk.inc.sd,
#         color = pm,
#         width = width)) +
#   scale_color_manual(name = "Parent material",
#                      values = c("andesite" = andesite,
#                                 "basalt" = basalt,
#                                 "granite" = granite)) +
#   scale_shape_manual(name = "Climate (year)",
#                      values = c("warm (2001)" = 15, 
#                                 "cool (2001)" = 16, 
#                                 "cold (2001)" = 17,
#                                 "warm (2019)" = 0, 
#                                 "cool (2019)" = 1, 
#                                 "cold (2019)" = 2)) +
#   facet_wrap(vars(min), scales = "free") +
#   ylab(expression('Bulk - Respired '*Delta*''^14*'C (‰)')) +
#   xlab(expression('Concentration (g kg'^-1*')')) +
#   theme_bw() +
#   theme(panel.grid.minor = element_blank())
```

```{r ts-stats}
# function for Tukey HSD tables
tukey.table.fx <- function(x, year, type, var) {
  depth <- paste0(unique(x$lyr_bot) - 10, "-", unique(x$lyr_bot), " cm")
  if (type == "inc") {
    x <- x[x$d14c > -200, c("d14c", var)]
  } 
  return(
    TukeyHSD(aov(reformulate(var, "d14c"), x))[var] %>%
    data.frame(.) %>%
    mutate(Pairs = rownames(.)) %>%
    mutate(across(where(is.numeric), round, 3)) %>%
    gt() %>%
    tab_header(
      title = depth,
      subtitle = paste(year, type, var)
    ))
}

### 2001
## bulk
sra.2001.bulk.df <- bind_rows(
  lapply(sra.19.01.rep.ls, function(ls) {
    ls <- lapply(ls, function(x) x[complete.cases(x)])
    d14c <- calc_14c(unlist(ls), 2001)
    df <- data.frame(d14c = d14c,
                     lyr_bot = rep(c(10, 20, 30), length(d14c) / 3))
    return(df)
  }),
  .id = "PMeco") %>%
  mutate(PM = substr(PMeco, 1, 2),
         ECO = substr(PMeco, 3, 4))
# PM
# lapply(split(sra.2001.bulk.df, sra.2001.bulk.df$lyr_bot), function(x) {
#   summary(lm(d14c ~ PM, x))
# })
lapply(split(sra.2001.bulk.df, sra.2001.bulk.df$lyr_bot), function(x) {
  tukey.table.fx(x, "2001", "bulk", "PM")
})
# ECO
# lapply(split(sra.2001.bulk.df, sra.2001.bulk.df$lyr_bot), function(x) {
#   summary(lm(d14c ~ ECO, x))
# })
lapply(split(sra.2001.bulk.df, sra.2001.bulk.df$lyr_bot), function(x) {
  tukey.table.fx(x, "2001", "bulk", "ECO")
})

## inc
sra.2001.inc.df2 <- cbind(sra.19.01.inc.df[rep(1:nrow(sra.19.01.inc.df), 2), c("PM", "ECO", "lyr_bot")],
                          d14c = c(sra.19.01.inc.df$d14c_min, sra.19.01.inc.df$d14c_max))
save(sra.2001.inc.df2, file = "sra.2001.inc.df2.RData")
# PM
# lapply(split(sra.2001.inc.df2, sra.2001.inc.df2$lyr_bot), function(x) {
#   summary(lm(d14c ~ PM, x[x$d14c > -200, ]))
# })
lapply(split(sra.2001.inc.df2, sra.2001.inc.df2$lyr_bot), function(x) {
  tukey.table.fx(x, "2001", "inc", "PM")
})
# ECO
# lapply(split(sra.2001.inc.df2, sra.2001.inc.df2$lyr_bot), function(x) {
#   summary(lm(d14c ~ ECO, x[x$d14c > -200, ]))
# })
lapply(split(sra.2001.inc.df2, sra.2001.inc.df2$lyr_bot), function(x) {
  tukey.table.fx(x, "2001", "inc", "ECO")
})

### 2019
## bulk
sra.2019.bulk.df <- bind_rows(sra.2019.ls)
# PM
# lapply(split(sra.2019.bulk.df, sra.2019.bulk.df$lyr_bot), function(x) {
#   if (nrow(x) == 27) summary(lm(d14c ~ PM, x[x$d14c > -200, ]))
# })
lapply(split(sra.2019.bulk.df, sra.2019.bulk.df$lyr_bot), function(x) {
  if (nrow(x) == 27) tukey.table.fx(x, "2019", "bulk", "PM")
})
# ECO
# lapply(split(sra.2019.bulk.df, sra.2019.bulk.df$lyr_bot), function(x) {
#   if (nrow(x) == 27) summary(lm(d14c ~ ECO, x[x$d14c > -200, ]))
# })
lapply(split(sra.2019.bulk.df, sra.2019.bulk.df$lyr_bot), function(x) {
  if (nrow(x) == 27) tukey.table.fx(x, "2019", "bulk", "ECO")
})
## inc
sra.2019.inc.df2 <- bind_rows(sra.2019.inc.ls)
save(sra.2019.inc.df2, file = "sra.2019.inc.df2.RData")
# PM
# lapply(split(sra.2019.inc.df2, sra.2019.inc.df2$lyr_bot), function(x) {
#   summary(lm(d14c ~ PM, x[x$d14c > -200, ]))
# })
lapply(split(sra.2019.inc.df2, sra.2019.inc.df2$lyr_bot), function(x) {
  tukey.table.fx(x, "2019", "inc", "PM")
})
# ECO
# lapply(split(sra.2019.inc.df2, sra.2019.inc.df2$lyr_bot), function(x) {
#   summary(lm(d14c ~ ECO, x[x$d14c > -200, ]))
# })
lapply(split(sra.2019.inc.df2, sra.2019.inc.df2$lyr_bot), function(x) {
  tukey.table.fx(x, "2019", "inc", "ECO")
})

# compare 2001 and 2019
# bulk
sra.01.19.bulk.df <- data.frame(
  rbind(sra.2001.bulk.df, 
        sra.2019.bulk.df[, which(names(sra.2019.bulk.df) %in% names(sra.2001.bulk.df))]),
  year = as.factor(c(rep(2001, nrow(sra.2001.bulk.df)), rep(2019, nrow(sra.2019.bulk.df))))) %>%
  filter(lyr_bot < 31)
sra.01.19.bulk.ls <- split(sra.01.19.bulk.df, sra.01.19.bulk.df$PMeco)
lapply(sra.01.19.bulk.ls, function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$PMeco), " 2001 vs. 2019"), "bulk", "year")
  })
})
# by PM
lapply(split(sra.01.19.bulk.df, sra.01.19.bulk.df$PM), function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$PM), " 2001 vs. 2019"), "bulk", "year")
  })
})
# by ECO
lapply(split(sra.01.19.bulk.df, sra.01.19.bulk.df$ECO), function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$ECO), " 2001 vs. 2019"), "bulk", "year")
  })
})
# inc
sra.01.19.inc.df <- data.frame(
  d14c = c(sra.19.01.inc[ , "d14c_min"],
           sra.19.01.inc[ , "d14c_max"]),
  sra.19.01.inc[ , c("PMeco", "lyr_bot", "PM", "ECO", "year")]) %>%
  mutate(year = as.factor(year))
sra.01.19.inc.ls <- split(sra.01.19.inc.df, sra.01.19.inc.df$PMeco)
lapply(sra.01.19.inc.ls, function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$PMeco), " 2001 vs. 2019"), "inc", "year")
  })
})
lapply(split(sra.01.19.inc.df, sra.01.19.inc.df$PM), function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$PM), " 2001 vs. 2019"), "inc", "year")
  })
})
lapply(split(sra.01.19.inc.df, sra.01.19.inc.df$ECO), function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    tukey.table.fx(x, paste0(unique(df$ECO), " 2001 vs. 2019"), "inc", "year")
  })
})
```

```{r min-14c-stats}
sra.01.19.min.reps <- left_join(
  merge(sra.01.19.bulk.df, sra.01.19.inc.df,
        by = c("PMeco", "PM", "ECO", "year", "lyr_bot"),
        suffixes = c("_blk", "_inc")),
  ras18.sp.df[ , c("Al_ox", "Al_py", "Fe_dc", "Fe_ox", "PM", "ECO", "lyr_bot")],
  by = c("PM", "ECO", "lyr_bot")) %>%
  mutate(Year = as.numeric(as.character(year)))

summary(lm(d14c_blk ~ Al_ox + lyr_bot + year, sra.01.19.min.reps))
sra.01.19.min.reps %>%
  mutate(eco = ifelse(ECO == "rf", "cold", ifelse(ECO == "wf", "cool", "warm"))) %>%
  mutate(ecoYear = paste0(eco, " (", year, ")")) %>%
  # filter(pmEcoDepth != "granitecold30" & pmEcoDepth != "granitecold20") %>%
  ggplot(., aes(Al_ox, d14c_blk)) +
  geom_point(aes(color = PM, shape = ecoYear), size = 3) +
  scale_color_manual(name = "Parent material",
                     values = c("AN" = andesite,
                                "BS" = basalt,
                                "GR" = granite),
                     labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite")) + 
  scale_shape_manual(name = "Climate (year)",
                     values = c("warm (2001)" = 15, 
                                "cool (2001)" = 16, 
                                "cold (2001)" = 17,
                                "warm (2019)" = 0, 
                                "cool (2019)" = 1, 
                                "cold (2019)" = 2)) +
  facet_wrap(vars(lyr_bot)) +
  ylab(expression('Bulk '*Delta*''^14*'C (‰)')) +
  xlab(expression('Oxalate extractable Al (g kg'^-1*')')) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```

```{r tukey-plots}
# color palettes for ECO & PM
warm <- "#BF812D"
cool <- "#80CDC1"
cold <- "#01665E"
granite <- "#9daba9"
andesite <- "#382dbf"
basalt <- "#bf382d"

# plot fx
boxplot.fx <- function(df, var, year, type, topsoil = FALSE, subsoil = FALSE) {
  atm <- ifelse(year == "2001", atm.d14.2001, atm.d14.2019)
  if (type == "inc") {
    df <- df[df$d14c > -200, ]
    ylim <- c(-65, 165)
  } else {
    if (topsoil) {
      df <- df[df$lyr_bot < 31, ]
      ylim <- c(-120, 165)
      }
    if (subsoil) {
      df <- df[df$lyr_bot > 31, ]
      ylim <- c(-270, 65)
    }
  }
  if (var == "PM") {
    df %>%
      mutate(pm = factor(ifelse(PM == "GR", "granite",
                                ifelse(PM == "AN", "andesite", "basalt")),
                         levels = c("granite", "andesite", "basalt"))) %>%
      group_by(pm, lyr_bot) %>%
      ggplot(., aes(pm, d14c)) +
      geom_hline(yintercept = atm, linetype = "dotted", alpha = 0.3) +
      geom_hline(yintercept = 0) +
      geom_boxplot(aes(color = pm), lwd = 1) +
      scale_color_manual(values = c("andesite" = andesite,
                                    "basalt" = basalt,
                                    "granite" = granite),
                         guide = "none") +
      scale_y_continuous(limits = ylim) +
      facet_grid(cols = vars(lyr_bot)) +
      ylab(expression(Delta*''^14*'C (‰)')) +
      ggtitle(paste(year, type)) +
      theme_bw() +
      theme(panel.grid = element_blank(),
            text = element_text(size = 14))
  } else {
    df %>%
      mutate(eco = factor(ifelse(ECO == "pp", "warm",
                                 ifelse(ECO == "wf", "cool", "cold")),
                          levels = c("warm", "cool", "cold"))) %>%
      group_by(eco, lyr_bot) %>%
      ggplot(., aes(eco, d14c)) +
      geom_hline(yintercept = atm, linetype = "dotted", alpha = 0.3) +
      geom_hline(yintercept = 0) +
      geom_boxplot(aes(color = eco), lwd = 1) +
      scale_color_manual(values = c("warm" = warm,
                                    "cool" = cool,
                                    "cold" = cold),
                         guide = "none") +
      scale_y_continuous(limits = ylim) +
      facet_grid(cols = vars(lyr_bot)) +
      ylab(expression(Delta*''^14*'C (‰)')) +
      ggtitle(paste(year, type)) +
      theme_bw() +
      theme(panel.grid = element_blank(),
            text = element_text(size = 14))
  }
}

# bulk
boxplot.fx(sra.2001.bulk.df, "PM", "2001", "bulk", topsoil = TRUE)
boxplot.fx(sra.2019.bulk.df, "PM", "2019", "bulk", topsoil = TRUE)
boxplot.fx(sra.2001.bulk.df, "ECO", "2001", "bulk", topsoil = TRUE)
boxplot.fx(sra.2019.bulk.df, "ECO", "2019", "bulk", topsoil = TRUE)
boxplot.fx(sra.2019.bulk.df, "ECO", "2019", "bulk", subsoil = TRUE)
# inc
boxplot.fx(sra.2001.inc.df2, "PM", "2001", "inc")
boxplot.fx(sra.2019.inc.df2, "PM", "2019", "inc")
boxplot.fx(sra.2001.inc.df2, "ECO", "2001", "inc")
boxplot.fx(sra.2019.inc.df2, "ECO", "2019", "inc")
```

```{r delta-delta-plots}
# data, unsummarized
sra.ts.all.raw <- rbind(
  sra.2001.bulk.df[ , names(sra.2001.bulk.df) %in% names(sra.2001.inc.df2)],
  sra.2019.bulk.df[ , names(sra.2019.bulk.df) %in% names(sra.2001.inc.df2)],
  sra.2001.inc.df2,
  sra.2019.inc.df2[ , names(sra.2019.inc.df2) %in% names(sra.2001.inc.df2)]) %>%
  mutate(eco = factor(ifelse(ECO == "pp", "warm",
                             ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = factor(ifelse(PM == "GR", "granite",
                                ifelse(PM == "AN", "andesite", "basalt")),
                         levels = c("granite", "andesite", "basalt")),
         Type = c(rep("bulk", length = nrow(sra.2001.bulk.df)),
                  rep("bulk", length = nrow(sra.2019.bulk.df)),
                  rep("inc", length = nrow(sra.2001.inc.df2)),
                  rep("inc", length = nrow(sra.2019.inc.df2))),
         year = c(rep(2001, length = nrow(sra.2001.bulk.df)),
                  rep(2019, length = nrow(sra.2019.bulk.df)),
                  rep(2001, length = nrow(sra.2001.inc.df2)),
                  rep(2019, length = nrow(sra.2019.inc.df2))))

# plot fx
ts.groupPlot.fx <- function(df, x, y) {
  quo_x <- sym(x)
  quo_y <- sym(y)
  if (x == "pm") {
    var.name <- "Parent material"
    var.values <- c("andesite" = andesite,
                    "basalt" = basalt,
                    "granite" = granite) 
  } else {
    var.name <- "Climate"
    var.values <-  c("warm" = warm,
                     "cool" = cool,
                     "cold" = cold)
  }
  plot.df <- df %>%
    filter(d14c > -200) %>%
    filter(lyr_bot < 31) %>%
    group_by(!! quo_x, lyr_bot, Type, year) %>%
    summarize(across(d14c, list(mean = mean, sd = sd)))
  if (y == "dd14c") {
    plot.df <- plot.df %>%
      mutate(atm = ifelse(year == 2001, atm.d14.2001, atm.d14.2019),
             dd14c = d14c_mean - atm,
             u = d14c_mean + d14c_sd - atm,
             l = d14c_mean - d14c_sd - atm)
    atm.df <- atm.14c
    atm.df$d14c <- 0
    ylab <- expression(Delta*Delta*''^14*'C (‰)') 
    } else {
      plot.df <- plot.df %>%
        mutate(u = d14c_mean + d14c_sd,
               l = d14c_mean - d14c_sd)
      atm.df <- atm.14c
      ylab <- expression(Delta*''^14*'C (‰)') 
    }
    ggplot(plot.df, aes(year, !! quo_y)) +
    geom_path(data = atm.df, aes(year, d14c)) +
    geom_path(aes(color = !! quo_x, linetype = Type), alpha = .5, lwd = 1) +
    geom_point(aes(color = !! quo_x), 
               size = 3, position = position_dodge(width = 1)) +
    geom_errorbar(
      aes(ymin = l,
          ymax = u,
          color = !! quo_x,
          alpha = Type),
      width = 1,
      position = position_dodge(width = 1)) +
    scale_color_manual(name = var.name,
                       values = var.values) +
    scale_fill_manual(name = "±SD",
                      values = var.values) +
    scale_alpha_manual(values = c("bulk" = 1,
                                  "inc" = .5)) +
    facet_grid(cols = vars(lyr_bot)) +
    ylab(ylab) +
    xlab("Year") +
    theme_bw() +
    theme(panel.grid = element_blank())
}
# plot
ts.groupPlot.fx(sra.ts.all.raw, "pm", "dd14c")
ts.groupPlot.fx(sra.ts.all.raw, "eco", "dd14c")
ts.groupPlot.fx(sra.ts.all.raw, "pm", "d14c_mean")
ts.groupPlot.fx(sra.ts.all.raw, "eco", "d14c_mean")
```
## Initial modeling

The goal of this modeling exercise is to see how parent material and climate/ecosystem affect estimates of soil carbon ages and transit times. Bulk soil ^14^C observations from 2001, 2009, and 2019 will be used to constrain the carbon models, as well as observations of ^14^C-CO~2~ from laboratory soil incubations of soils collected in 2001 and 2019. Previous work has indicated that the carbon stocks at these sites is likely at equilibrium, so we will apply the steady-state assumption to the modeling.

### Two-pool models

One pool models have been shown repeatedly to be inadequate for describing soil carbon dynamics. However, as simple models are easier to constrain, we will start with a two-pool parallel and two-series models, as these are the simplest model system beyond the single pool approach. 

The two-pool parallel model requires the following parameters:
* decomposition constants for each pool (*k*~1~, *k*~2~)
* input partitioning coefficient ($\gamma$)
* steady-state carbon stocks (C)
* inputs (I)
* initial values of ^14^C
1
The two-pool series model requires the following parameters:
* decomposition constants for each pool (*k*~1~, *k*~2~)
* transfer coefficient ($\alpha$)
* steady-state carbon stocks (C)
* inputs (I)
* initial values of ^14^C

Decomposition rates (*k*) are related to the amount of ^14^C in a pre-bomb system (fraction modern, *F*) at steady-state by the following equations (cf. Schuur, Druffle, and Trumbore, 2016):
>**Eq. 1**

$$F = \frac{k}{k + \lambda}$$
>**Eq. 2**

$$k = \frac{\lambda \cdot F}{1 - F}$$
>where $\lambda$ is the radioactive decay constant (1/8267).

As the decomposition rates will vary, the initial ^14^C content can be determined dynamically with equation 1.

Carbon stocks are known, while inputs will be estimated and are related to the steady-state conditions by the following equation: 
>**Eq. 3**

$$I = (k_{1} \cdot C_{1}) + (k_{2} \cdot C_{2})$$
>where *C~1~* and *C~2~* are the carbon stocks of the two model pools.

Both stocks and inputs can be scaled to the known value of the total carbon pool once the steady-state parameters (*k~1~*, *k~2~*, and $\gamma$ or $\alpha$) have been determined. Pool sizes are a function of the inputs and input partitioning coefficient at steady-state.

A Monte-Carlo Markov chain approach will be used for parameter estimation in combination with an initial optimization algorithm to determine the best set of initial parameters.

## Workflow

Initial model fitting was performed for both model structures using generous parameter ranges [0, 1] for all three parameters (*k~1~*, *k~2~*, $\gamma$ or $\alpha$). The initial parameter set was found by fitting the models by eye, followed by optimization with the function "modFit" (R package FME), using the Nelder-Mead algorithm. The best set of parameters found by modFit was then used as the input to a Monte Carlo Markov Chain (MCMC), using the function "modMCMC" (R package FME). The number of iterations for the MCMC optimization was set at 5000 intially, with delayed rejection employed to increase efficiency. 

The sum of the mean squared error for the best parameter set was slightly lower for the parallel structure than for the series structure. Additionally, the overall mean error of the residuals was also lower for the parallel structure, moderately so for the bulk C observations but substantially so for the respiration observations (in andesite and granite soils in particular).

However, these initial fits yielded unrealistic parameter estimates for multiple sites, particularly at the lower depths. Additionally, the modFit output showed very high correlation between the parameters for both model structures (slightly higher for the two-pool series model). 

```{r mod-utils}
# k from fraction modern
k <- function (Fm) {
  (Fm * lambda)/(1 - Fm)
}

# d14C from fraction modern 
fm_14c <- function (fm, date) {
  (fm * exp(lambda * (1950 - date)) - 1) * 1000
}

# pre-bomb fraction modern from k (steady-state assumed)
fm <- function (k){
  k/(k + lambda)
}
```

```{r mod-constraints, include = FALSE}
# Indices for each depth increment
ix.10 <- seq(1, 27, 3)
ix.20 <- seq(2, 27, 3)
ix.30 <- seq(3, 27, 3)

## SOC stocks
# use 2019 SOC stocks for steady-state estimates
csoc.19.0_30.df <- bind_rows(
  lapply(sra.2019.sp.ls, function(df) {
    df <- suppressMessages(
      df %>%
        filter(lyr_bot < 31 & lyr_bot > 0) %>%
        select(PMeco, lyr_top, lyr_bot, lyr_soc) %>%
        group_by(PMeco, lyr_top, lyr_bot))
    return(data.frame(df))
  })
)
# write.csv(csoc.19.0_30.df, "csoc.19.0_30.csv")

# convert to 27 element list
csoc.19.0_30.ls <- split(csoc.19.0_30.df, paste0(csoc.19.0_30.df$PMeco, "_", csoc.19.0_30.df$lyr_top, "-", csoc.19.0_30.df$lyr_bot))

# average
csoc.19.0_30 <- lapply(csoc.19.0_30.ls, function(df) {
  data.frame(
    df %>%
      group_by(PMeco, lyr_top, lyr_bot) %>%
      summarize(lyr_soc = mean(lyr_soc)))
})

# make into obs data frame for mod.cost fx
obs.cStock <- lapply(csoc.19.0_30.ls, function(df) {
  return(data.frame(time = rep(c(2001.5, 2009.5, 2019.5), each = 3), cStock = rep(df$lyr_soc, 3)))
})

## Inputs
# initial inputs will be set at 4% of the layer carbon stocks (arbitrary)
in.i <- lapply(csoc.19.0_30, function(x) .04 * x$lyr_soc)
# Inputs will be adjusted based on the fitted parameters to match measured stocks later

## 14C constraints
# bulk
obs.bulk.14c <- unlist(
  lapply(seq_along(sra.19.01.rep.ls), function(i) {
  # index along depth intervals 0-10, 10-20, 20-30
  depth.ls <- lapply(seq_along(1:3), function(j) {
    c(unlist(lapply(sra.19.01.rep.ls[[i]], "[[", j)),
      split(split(sra.19.01.09, sra.19.01.09$PMeco)[[i]],
            split(sra.19.01.09, sra.19.01.09$PMeco)[[i]]["lyr_bot"])[[j]][ , "fm"][2],
      unlist(split(sra.2019.ls[[i]], sra.2019.ls[[i]]["lyr_bot"])[[j]]["fm"]))
  })
  reps01 <- length(sra.19.01.rep.ls[[i]])
  depth.dfs <- lapply(depth.ls, function(fm) {
    data.frame(time = c(rep(2001.5, reps01), 2009.5, rep(2019.5, 3)),
               bulkC = Delta14C_from_AbsoluteFractionModern(fm))
  })
  return(depth.dfs)
}), recursive = FALSE)
names(obs.bulk.14c) <- paste0(rep(c("AN", "BS", "GR"), each = 9),
                              rep(c("pp", "rf", "wf"), each = 3, times = 3),
                              rep(c("_0-10", "_10-20", "_20-30"), times = 9))
save(obs.bulk.14c, file = "obs.bulk.14c.RData")

# respiration
sra.19.01.inc.min.max <- unlist(
  lapply(seq_along(1:3), function(i) {
    lapply(
      mapply(merge,
             lapply(lapply(sra.19.01.inc.ls, "[[", 2), "[[", i), 
             lapply(lapply(sra.19.01.inc.ls, "[[", 3), "[[", i),
             SIMPLIFY = FALSE),
      function(df) {
        data.frame(time = 2001.5, resp = c(df$x, df$y))
      })
  }), recursive = FALSE)
names(sra.19.01.inc.min.max) <- paste0(names(sra.19.01.inc.min.max), 
                                       rep(c("_0-10", "_10-20", "_20-30"), each = 9))
sra.19.01.inc.min.max <- lapply(sra.19.01.inc.min.max, function(df) {
  df$resp <- calc_14c(df$resp, 2001)
  return(df)
})

sra.2019.inc.min.max <- unlist(lapply(sra.2019.inc.ls, function(df) {
  lapply(split(df, df$lyr_bot), function(x) {
    x$resp <- x$d14c
    x$time <- x$Year + .5
    return(x[ , c("time", "resp")])
  })
}), recursive = FALSE)
names(sra.2019.inc.min.max) <- gsub("\\.", "_", names(sra.2019.inc.min.max))
for (i in seq_along(names(sra.2019.inc.min.max))) {
  names(sra.2019.inc.min.max)[i] <- ifelse(grepl("10", names(sra.2019.inc.min.max)[i]),
                                   gsub("10", "0-10", names(sra.2019.inc.min.max)[i]),
                                   ifelse(grepl("20", names(sra.2019.inc.min.max)[i]),
                                          gsub("20", "10-20", names(sra.2019.inc.min.max)[i]), gsub("30", "20-30", names(sra.2019.inc.min.max)[i])))
}

obs.resp.14c <- rbind(bind_rows(sra.19.01.inc.min.max, .id = "id"),
                      bind_rows(sra.2019.inc.min.max, .id = "id"))
obs.resp.14c <- lapply(split(obs.resp.14c, obs.resp.14c$id), function(df) df[ , 2:3])

save(obs.resp.14c, file = "obs.resp.14c.RData")

## input/stock ratio
ras06 <- data.frame(read_excel("../data/external/sra_ras_inc/RespRates_Rasmussen2006.xlsx", sheet = "RatesSum"))
# Obs cost
obs.flx.stock <- lapply(split(ras06, ras06$PMeco), function(x) {
  data.frame(time = 2001.5, # arbitrary
             flx.stock = x[ , "flx_stock_ratio"])
})

# calculate inputs using flx/stock ratio
in.flx.stock <- lapply(seq_along(obs.flx.stock), function(i) {
  obs.flx.stock[[i]][["flx.stock"]]/csoc.19.0_30[ix.10][[i]][["lyr_soc"]]
})

# Flux estimated from Goulden et al. 2012; Tang et al. 2005; Wang et al. 2000; Gaudinski 2000
# fluxes by elevation from GPP reported in Goulden et al. Fig. 5 and approximated
# Rh percentage from Tang et al. 2005 = 0.44 (ann. mean Blodgett); cf. 0.48 Harvard Forest
# A horizon est. 0.55 from Gaudinski
# assuming A = 0-30, assume 0-10 = 50%, 10-20 = 30%, 20-30 = 20% of total A production 
hznA.Rh.kgm2 <- 0.44 * 0.55 * 10^-3
gpp.ls <- c(1800, 1600, 1400)
in.frc.ls <- c(0.5, 0.3, 0.2)

# fx for calculating inputs
in.flx.fx <- function(PMeco_depth) {
  gpp <- ifelse(grepl("pp", PMeco_depth), gpp.ls[1], ifelse(grepl("wf", PMeco_depth), gpp.ls[2], gpp.ls[3]))
  in.frc <- ifelse(grepl("0-10", PMeco_depth), in.frc.ls[1], ifelse(grepl("10-20", PMeco_depth), in.frc.ls[2], in.frc.ls[3]))
  return(gpp * in.frc * hznA.Rh.kgm2)
}

# input list
in.est <- lapply(seq_along(obs.cStock), function(i) {
  PMeco_depth <- names(obs.cStock)[i]
  return(in.flx.fx(PMeco_depth))
})
names(in.est) <- names(obs.cStock)
```

```{r mod-funs-gen}
# index of years for which bulk/resp 14C are known
year.ix <- c(which(Datm$Date == 2001.5),
             which(Datm$Date == 2009.5),
             which(Datm$Date == 2019.5))

# function for saving constraint data in a dataframe for plotting in ggplot'
con.df.fx <- function(PMeco_depth) {
  bulk.df <- obs.bulk.14c[[PMeco_depth]]
  resp.df <- obs.resp.14c[[PMeco_depth]]
  return(
    con.df <- data.frame(pool = c(rep("bulk C", nrow(bulk.df)), rep("respiration", nrow(resp.df))),
                         d14c = c(bulk.df$bulkC, resp.df$resp),
                         Year = c(bulk.df$time, resp.df$time)))
}

# plot function
C14.2p.plot.fx <- function(plot.df, con.df, mod, PMeco_depth) {
  plot.df %>%
  filter(pool == "bulk C" | pool == "respiration" | pool == "atm") %>%
  ggplot(., aes(years, d14C, color = pool)) +
  geom_path() +
  geom_point(data = con.df, aes(Year, d14c, color = pool), size = 3) +
  scale_color_manual(
    name = "Pool",
    values = c("atm" = 8,
               "bulk C" = "black",
               "fast" = "#D81B60",
               "slow" = "#1E88E5",
               "respiration" = "#FFC107")) +
  scale_x_continuous(limits = c(1950, 2022)) +
  ggtitle(paste(PMeco_depth, mod)) +
  xlab("Year") +
  ylab(expression(''*Delta*''^14*'C (‰)')) +
  theme_bw() +
  theme(panel.grid = element_blank())
}
C14.1p.plot.fx <- function(plot.df, con.df, mod, PMeco_depth) {
  ggplot(plot.df, aes(years, d14C, color = pool)) +
  geom_path() +
  geom_point(data = con.df, aes(Year, d14c, color = pool), size = 3) +
  scale_color_manual(
    name = "Pool",
    values = c("atm" = 8,
               "bulk C" = "black",
               "respiration" = "#FFC107")) +
  scale_x_continuous(limits = c(1950, 2022)) +
  ggtitle(paste(PMeco_depth, " 1p bulk + 1p resp")) +
  xlab("Year") +
  ylab(expression(''*Delta*''^14*'C (‰)')) +
  theme_bw() +
  theme(panel.grid = element_blank())
}

# set up model function for optimization
# NOTE: par[3] for 2ps model changed to proportion transferred (no longer = a21)
# therefore, a21 = par[3] * par[1]
modFun_2p <- function(pars, In, lag = 0, pass = TRUE, out = "modFit", mod) {
 
  # intial 14C
  F0_Delta14C <- unlist(lapply(pars[1:2], function(x) Delta14C_from_AbsoluteFractionModern(fm(x))))
  
  # model matrix
  A <- -diag(pars[1:2])
  if (mod == "2ps") {
    a21 <- pars[3] * pars[1]
    A[2, 1] <- a21
  }
    
  # steady-state C stocks
  if (mod == "2pp") {
    ss.cstock <- (-1 * solve(A) %*% c(In * pars[3], In * (1 - pars[3])))
  } else {
    ss.cstock <- (-1 * solve(A) %*% c(In, 0))
  }
  
  # time index
  ix.t <- c((lag + 1):nrow(Datm))
  
  # model
  if (mod == "2pp") {
    mod <- TwopParallelModel14(t = Datm$Date[ix.t],
                               ks = pars[1:2],
                               C0 = c(ss.cstock[1], ss.cstock[2]),
                               F0_Delta14C = F0_Delta14C,
                               In = In,
                               gam = pars[3],
                               inputFc = Datm,
                               lag = lag,
                               pass = pass)
  } else {
    mod <- TwopSeriesModel14(t = Datm$Date[ix.t],
                             ks = pars[1:2],
                             C0 = c(ss.cstock[1], ss.cstock[2]),
                             F0_Delta14C = F0_Delta14C,
                             In = In,
                             a21 = a21,
                             inputFc = Datm,
                             lag = lag,
                             pass = pass)
  }
  
  # get mod values
  C14m <- getF14C(mod)
  C14p <- getF14(mod)
  C14r <- getF14R(mod)
  Ctot <- getC(mod)
  
  if(out == "modFit") {
    # dataframe for modFit fx
    return(data.frame(
      time = Datm$Date[ix.t],
      bulkC = C14m, 
      resp = C14r,
      cStock = rowSums(Ctot)))
  } else {
    # data frame for plotting
    return(data.frame(
      years = rep(Datm$Date[ix.t], 5),
      d14C = c(C14p[,1], 
               C14p[,2], 
               C14m,
               C14r,
               Datm$NHc14[ix.t]),
      pool = rep(c("fast", "slow", "bulk C", "respiration", "atm"), each = nrow(C14p))))
  }
}

# 1p modFun
modFun_1p <- function(pars, In, lag = 0, out = "modFit", mod, pass = TRUE) {
 
  # intial 14C
  F0_Delta14C <- Delta14C_from_AbsoluteFractionModern(fm(pars))
  
  # steady-state C stocks
  ss.cstock <- In/pars
  
  # time index
  ix.t <- c((lag + 1):nrow(Datm))
  
  # model
  mod <- suppressWarnings(
    # warnings suppressed due to the "Fc" warning
    OnepModel14(t = Datm$Date[ix.t],
                     k = pars,
                     C0 = ss.cstock,
                     F0_Delta14C = F0_Delta14C,
                     In = In,
                     inputFc = Datm,
                     lag = lag,
                     pass = pass)
  )
  
  # get mod values
  C14m <- getF14C(mod)
  Ctot <- getC(mod)
  
  if(out == "modFit") {
    # dataframe for modFit fx
    return(data.frame(
      time = Datm$Date[ix.t],
      bulkC = C14m,
      cStock = Ctot))
  } else {
    # data frame for plotting
    return(data.frame(
      years = rep(Datm$Date[ix.t], 1),
      d14C = c(C14m,
               Datm$NHc14[ix.t]),
      pool = rep(c("bulk C", "atm"), each = length(C14m))))
  }
}

# function for trial and error approach to finding initial parameter set
par.fx <- function(pars, In, lag = 0, out = "plot.df", verbose = TRUE, mod, pass = FALSE) {
  
  # model matrix
  A <- -diag(pars[1:2])
  if (mod == "2ps") {
    a21 <- pars[3] * pars[1]
    A[2, 1] <- a21
    # steady-state stocks
    ss.cstock <- round((-1 * solve(A) %*% c(In, 0)), 1)
  } else if (mod == "2pp") {
    # steady-state stocks
    ss.cstock <- round((-1 * solve(A) %*% c(In * pars[3], In * (1 - pars[3]))), 1)
  } else {
    ss.cstock <- In/pars
  }
  
  cstock.sum <- ifelse(length(ss.cstock) == 1, ss.cstock, colSums(ss.cstock))
  
  # print site and steady-state stocks
  if (verbose) {
    cat(paste0(PMeco_depth, "\n"))
    if (mod == "2ps" | mod == "2pp") {
      cat(paste0(ss.cstock[1], " (fast pool)\n", ss.cstock[2], " (slow pool)\n"))
      cat(paste0("slow pool: ", round(ss.cstock[2] / cstock.sum * 100, 0), "%\n")) 
    }
    cat(round(cstock.sum, 1), " (modeled stocks)\n")
    cat(round(csoc.19.0_30[[PMeco_depth]][ , "lyr_soc"], 1), " (measured stocks)\n") 
  }
  if (mod == "1p") {
    return(modFun_1p(pars = pars, In = In, lag = lag, out = out, mod = "1p", pass = pass))
  }
  if (mod == "2pp") {
   return(modFun_2p(pars = pars, In = In, lag = lag, out = out, mod = "2pp", pass = pass)) 
  } else if (mod == "2ps") {
    return(modFun_2p(pars = pars, In = In, lag = lag, out = out, mod = "2ps", pass = pass)) 
  }
}
```

```{r inputs-stocks}
## adjust inputs to match stocks
# function for calculating steady-state SOC stocks
soc.fx <- function(modStr, pars, In, out = "pools") {
  if (modStr == "2pp") {
    cmat <- -1 * solve(-diag(pars[1:2])) %*% c(In * pars[3], In * (1 - pars[3]))
  } else {
    A <- -diag(pars[1:2])
    A[2, 1] <- pars[3] # note that a21 defined as pct transfer * k1
    cmat <- -1 * solve(A) %*% c(In, 0) # In is total input into the system
  }
  if (out == "pools") {
    return(cmat)
  } else {
    return(colSums(cmat))
  }
}

in.fit.fx <- function(modStr, pars, initialIn, SOC) {
  # sequence of possible input values
  if  (SOC < soc.fx(modStr, pars, initialIn, out = "sum")) {
    ins <- seq(.01, 
               initialIn, 
               .01)
    } else {
      ins <- seq(initialIn, 
                 SOC, 
                 .01)
    }
  # modeled stocks
  soc_mod <- lapply(seq_along(ins), function(j) {
    soc.fx(modStr, pars, ins[j], out = "sum")
  })
  ix <- which.min(abs(unlist(soc_mod) - SOC))
  return(ins[ix])
}

# load initial parameter set
load("../data/derived/modFit_pars/pars.i.2pp_2021-03-30.Rdata")
load("../data/derived/modFit_pars/pars.i.2ps_2020-11-16.Rdata")

## inputs for initial par set and measured stocks
# 2pp
in.meas.2pp <- lapply(seq_along(pars.i.2pp[ix.10]), function(i) {
  PMeco_depth <- names(pars.i.2pp[ix.10])[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.i.2pp[ix.10][[i]], in.i[ix.10][[i]], SOC))
})
names(in.meas.2pp) <- names(pars.i.2pp[ix.10])
# 2ps
in.meas.2ps <- lapply(seq_along(pars.i.2ps[ix.10]), function(i) {
  PMeco_depth <- names(pars.i.2ps[ix.10])[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.i.2ps[ix.10][[i]], in.i[ix.10][[i]], SOC))
})
names(in.meas.2ps) <- names(pars.i.2ps[ix.10])

# Flux estimated from Goulden et al. 2012; Tang et al. 2005; Wang et al. 2000; Gaudinski 2000
# fluxes by elevation from GPP reported in Goulden et al. Fig. 5 and approximated
# Rh percentage from Tang et al. 2005 = 0.44 (ann. mean Blodgett); cf. 0.48 Harvard Forest
# A horizon est. 0.55 from Gaudinski
# assuming A = 0-30, assume 0-10 = 50%, 10-20 = 30%, 20-30 = 20% of total A production 
hznA.Rh.kgm2 <- 0.44 * 0.55 * 10^-3
gpp.ls <- c(1800, 1600, 1400)
in.frc.ls <- c(0.5, 0.3, 0.2)

# fx for calculating inputs
in.flx.fx <- function(PMeco_depth) {
  gpp <- ifelse(grepl("pp", PMeco_depth), gpp.ls[1], ifelse(grepl("wf", PMeco_depth), gpp.ls[2], gpp.ls[3]))
  in.frc <- ifelse(grepl("0-10", PMeco_depth), in.frc.ls[1], ifelse(grepl("10-20", PMeco_depth), in.frc.ls[2], in.frc.ls[3]))
  return(gpp * in.frc * hznA.Rh.kgm2)
}

# input list
in.est <- lapply(seq_along(pars.i.2pp), function(i) {
  PMeco_depth <- names(pars.i.2pp)[i]
  return(in.flx.fx(PMeco_depth))
})
names(in.est) <- names(pars.i.2pp)
```
## Parameter optimization

Optimizing the parameter set requires imposing costs and optionally constraining the allowable range of values for each parameter. Given that we only have data for three time points, this is a relatively sparse data set for constraining these models. Accordingly, the optimization procedure will benefit from *a priori* constraints of the allowable parameter ranges. For example, since we assume that the system cannot be adequately modeled as a single homogenous reservoir, we will ensure that the optimization procedure cannot collapse the two-pool system into a single pool. This can be mitigated in the two-pool parallel optimization by constraining $\gamma$ (i.e. the percentage of the inputs entering the fast pool) to a range of 50% to 95%. Similarly, for the two-pool series model structure we can constrain the range of the transfer coefficient to be between 0.0 and 0.1, ensuring that some carbon remains in the fast cycling pool.

Additionally, to enforce a relatively fast cycling pool and relatively slower cycling pool, we will loosely constrain the intrinsic decomposition rates as well (both model structures):

*k~1~*: [0.02, 1.00] (50 to 1 year)
*k~2~*: [0.0001, 0.02] (10,000 to 50 years)

Finally, the models will be run to enforce steady-state, i.e. with unvarying carbon stocks. The amount of carbon observed in the system will be used in the cost function in addition to the radiocarbon observations made in 2001, 2009, and 2019. The inputs will be estimated from net ecosystem exchange (NEE) data measured at nearby eddy covariance sites: Blodgett experimental forest (AmeriFlux), Lower Teakettle (NEON), and Soaproot Saddle (NEON). Alternatively, using correlations between fluxes measured from these eddy covariance towers and GPP estimated from satellite retrievals of SIF, estimates can be made for inputs at the pixels corresponding to each site location.

```{r opt-mod, eval = FALSE}
# Note: this only runs if eval flag switched to TRUE
## Optimize model pars
# Cost function (evaluates error as model vs. obsv, per FME req)
# note that we have to set "pass" to TRUE so SoilR model doesn't fail (neg. resp)
mod.fits.fx <- function(mod, pars, In, sub, lag = 0, upper, lower, cost) {
  
  # start loop
  lapply(seq_along(pars[sub]), function(i) {
    
    # start timer and print PMeco_depth
    start <- Sys.time()
    cat(paste0(names(pars)[sub][i], " parameter fitting\n"))
  
    # define pars
    pars <- pars[sub][[i]]
    if (mod == "2pp") {
      names(pars) <- c("k1", "k2", "gam")
    } else {
      names(pars) <- c("k1", "k2", "tc")
    }
    
    # Set input
    In <- In[sub][[i]]
    
    # define cost function
    if (cost == "14C + cStock") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        cost1 <- modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE)
        cost2 <- modCost(model = modelOutput, obs = obs.resp.14c[sub][[i]], scaleVar = TRUE, cost = cost1) 
        return(modCost(model = modelOutput, obs = obs.cStock[sub][[i]], cost = cost2))
      }
    } else if (cost == "14C + stock/flx") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        cost1 <- modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE)
        cost2 <- modCost(model = modelOutput, obs = obs.resp.14c[sub][[i]], scaleVar = TRUE, cost = cost1) 
        return(modCost(model = modelOutput, obs = obs.flx.stock[[i]], cost = cost2))
      }
    } else if (cost == "14C") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        cost1 <- modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE)
        return(modCost(model = modelOutput, obs = obs.resp.14c[sub][[i]], scaleVar = TRUE, cost = cost1))
      } 
    } else if (cost == "14C bulk + cStock") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        cost1 <- modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE)
        return(modCost(model = modelOutput, obs = obs.cStock[sub][[i]], cost = cost1))
      }
    } else if (cost == "14C bulk only") {
      mod.Cost <- function(pars) {
        modelOutput <- modFun_2p(pars, In, mod = mod, lag = lag)
        return(modCost(model = modelOutput, obs = obs.bulk.14c[sub][[i]], scaleVar = TRUE))
      }
    }
    
    # fit pars
    fit <- tryCatch(
      modFit(f = mod.Cost,
             p = pars,
             method = 'Nelder-Mead',
             upper = upper, 
             lower = lower),
      error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
    
    Sfun <- sensFun(mod.Cost, fit$par)
    
    # End timer and print elapsed time
    end <- Sys.time()
    cat(paste0("time: ", end - start, "\n"))
    
    # Return fitted parameters and sensitivity
    return(list(modfit = fit, sens = Sfun))
  }) 
}

## 2pp
# par range [0, 1] for all pars
mod.sens.fits.2pp <- mod.fits.fx(mod = "2pp",
                                 pars = pars.i.2pp,
                                 In = in.i,
                                 sub = ix.10,
                                 upper = c(1, 1, 1),
                                 lower = c(0, 0, 0),
                                 cost = "14C")
names(mod.sens.fits.2pp) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp <- lapply(mod.sens.fits.2pp, function(x) x[[1]])
# constrain gamma to [0.5, 0.95]
mod.sens.fits.2pp.p3.5.95 <- mod.fits.fx(mod = "2pp",
                                    pars = pars.i.2pp,
                                    sub = ix.10,
                                    In = in.i,
                                    upper = c(1, 1, 0.951),
                                    lower = c(0, 0, 0.5))
names(mod.sens.fits.2pp.p3.5.95) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp.p3.5.95, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp.p3.5.95", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp.p3.5.95 <- lapply(mod.sens.fits.2pp.p3.5.95, function(x) x[[1]])

# 2pp3 (par range constraints, inputs fit to meas stocks)
mod.sens.fits.2pp3 <- mod.fits.fx(mod = "2pp",
                                  pars = pars.i.2pp,
                                  sub = ix.10,
                                  In = in.meas.2pp,
                                  upper = c(1, .02, .951),
                                  lower = c(.04, .0001, .5),
                                  cost = "14C only")
names(mod.sens.fits.2pp3) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp3, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp3", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp3 <- lapply(mod.sens.fits.2pp3, function(x) x[[1]])
# 2pp3s (par range constraints, inputs fit to meas stocks, + stock constraint)
mod.sens.fits.2pp3s <- mod.fits.fx(mod = "2pp",
                                   pars = pars.i.2pp,
                                   sub = ix.10,
                                   In = in.meas.2pp,
                                   upper = c(1, .02, .951),
                                   lower = c(.04, .0001, .5),
                                   cost = "cStock")
names(mod.sens.fits.2pp3s) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp3s, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp3s", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp3s <- lapply(mod.sens.fits.2pp3s, function(x) x[[1]])

## 2ps
# par range [0, 1] for all pars
mod.sens.fits.2ps <- mod.fits.fx(mod = "2ps",
                                 pars = pars.i.2ps, 
                                 sub = ix.10,
                                 In = in.i,
                                 upper = c(1, 1, 1),
                                 lower = c(0, 0, 0),
                                 cost = "14C")
names(mod.sens.fits.2ps) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps <- lapply(mod.sens.fits.2ps, function(x) x[[1]])

# par range [0, 1] for all pars, stocks + 14C, w/ estimated inputs
# 10
mod.sens.fits.2ps5.10 <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps, 
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, 1, 1),
                                     lower = c(0, 0, 0),
                                     cost = "14C + cStock")
names(mod.sens.fits.2ps5.10) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps5.10, file = paste0("../data/derived/modFit_pars/", "mod.sens.fits.2ps5.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps5.10 <- lapply(mod.sens.fits.2ps5.10, function(x) x[[1]])
# 20
mod.sens.fits.2ps5.20 <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps, 
                                     sub = ix.20,
                                     In = in.est,
                                     upper = c(1, 1, 1),
                                     lower = c(0, 0, 0),
                                     cost = "14C + cStock")
names(mod.sens.fits.2ps5.20) <- names(pars.i.2ps)[ix.20]
save(mod.sens.fits.2ps5.20, file = paste0("../data/derived/modFit_pars/", "mod.sens.fits.2ps5.20", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps5.20 <- lapply(mod.sens.fits.2ps5.20, function(x) x[[1]])

# 20-30
mod.sens.fits.2ps.30 <- mod.fits.fx(mod = "2ps",
                                    pars = pars.i.2ps, 
                                    sub = ix.30,
                                    In = in.i,
                                    upper = c(1, 1, 1),
                                    lower = c(0, 0, 0),
                                    cost = "14C")
names(mod.sens.fits.2ps.30) <- names(pars.i.2ps)[ix.30]
save(mod.sens.fits.2ps.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps.30 <- lapply(mod.sens.fits.2ps.30, function(x) x[[1]])

# 2ps3 (par range constraints, inputs fit to meas stocks)
mod.sens.fits.2ps3 <- mod.fits.fx(mod = "2ps",
                                  pars = pars.i.2ps,
                                  sub = ix.10,
                                  In = in.meas.2ps,
                                  upper = c(1, 1, .15),
                                  lower = c(0, 0, .0004),
                                  cost = "14C only")
names(mod.sens.fits.2ps3) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps3, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps3", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps3 <- lapply(mod.sens.fits.2ps3, function(x) x[[1]])
# 2ps3 (par range constraints, inputs fit to meas stocks, + stock constraint)
mod.sens.fits.2ps3s <- mod.fits.fx(mod = "2ps",
                                   pars = pars.i.2ps,
                                   sub = ix.10,
                                   In = in.meas.2ps,
                                   upper = c(1, .02, .1),
                                   lower = c(.04, .0001, 0),
                                   cost = "cStock")
names(mod.sens.fits.2ps3s) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps3s, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps3s", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps3s <- lapply(mod.sens.fits.2ps3s, function(x) x[[1]])

### 2p4 (par range set, stock and bulk 14C costs, GPP-based inputs by eco)
## 2pp
# 0-10
mod.sens.fits.2pp4.10 <- mod.fits.fx(mod = "2pp",
                                     pars = pars.i.2pp,
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, .02, .951),
                                     lower = c(.04, .0001, .5),
                                     cost = "14C bulk + cStock")
names(mod.sens.fits.2pp4.10) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp4.10, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp4.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp4.10 <- lapply(mod.sens.fits.2pp4.10, function(x) x[[1]])
# 20-30
mod.sens.fits.2pp4.30 <- mod.fits.fx(mod = "2pp",
                                     pars = pars.i.2pp,
                                     sub = ix.30,
                                     In = in.est,
                                     upper = c(1, .02, .951),
                                     lower = c(.04, .0001, .5),
                                     cost = "14C bulk + cStock")
names(mod.sens.fits.2pp4.30) <- names(pars.i.2pp)[ix.30]
save(mod.sens.fits.2pp4.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp4.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp4.30 <- lapply(mod.sens.fits.2pp4.30, function(x) x[[1]])
## 2ps
# 0-10
mod.sens.fits.2ps4.10 <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps,
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, .02, .1),
                                     lower = c(.04, .0001, 0),
                                     cost = "14C bulk + cStock")
names(mod.sens.fits.2ps4.10) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps4.10, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps4.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps4.10 <- lapply(mod.sens.fits.2ps4.10, function(x) x[[1]])
# 20-30
mod.sens.fits.2ps4.30 <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps,
                                     sub = ix.30,
                                     In = in.est,
                                     upper = c(1, .02, .1),
                                     lower = c(.04, .0001, 0),
                                     cost = "14C bulk + cStock")
names(mod.sens.fits.2ps4.30) <- names(pars.i.2ps)[ix.30]
save(mod.sens.fits.2ps4.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps4.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps4.30 <- lapply(mod.sens.fits.2ps4.30, function(x) x[[1]])

### 2p4r (par range set, stock, bulk, and respiration 14C costs, GPP-based inputs by eco)
## 2pp
# 0-10
mod.sens.fits.2pp4r.10 <- mod.fits.fx(mod = "2pp",
                                      pars = pars.i.2pp,
                                      sub = ix.10,
                                      In = in.est,
                                      upper = c(1, .02, .951),
                                      lower = c(.04, .0001, .5),
                                      cost = "14C + cStock")
names(mod.sens.fits.2pp4r.10) <- names(pars.i.2pp)[ix.10]
save(mod.sens.fits.2pp4r.10, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp4r.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp4r.10 <- lapply(mod.sens.fits.2pp4r.10, function(x) x[[1]])
# 20-30
mod.sens.fits.2pp4r.30 <- mod.fits.fx(mod = "2pp",
                                      pars = pars.i.2pp,
                                      sub = ix.30,
                                      In = in.est,
                                      upper = c(1, .02, .951),
                                      lower = c(.04, .0001, .5),
                                      cost = "14C + cStock")
names(mod.sens.fits.2pp4r.30) <- names(pars.i.2pp)[ix.30]
save(mod.sens.fits.2pp4r.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2pp4r.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2pp4r.30 <- lapply(mod.sens.fits.2pp4r.30, function(x) x[[1]])
## 2ps
# 0-10
mod.sens.fits.2ps4r.10 <- mod.fits.fx(mod = "2ps",
                                      pars = pars.i.2ps,
                                      sub = ix.10,
                                      In = in.est,
                                      upper = c(1, .02, .1),
                                      lower = c(.04, .0001, 0),
                                      cost = "14C + cStock")
names(mod.sens.fits.2ps4r.10) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps4r.10, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps4r.10", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps4r.10 <- lapply(mod.sens.fits.2ps4r.10, function(x) x[[1]])
# 20-30
mod.sens.fits.2ps4r.30 <- mod.fits.fx(mod = "2ps",
                                      pars = pars.i.2ps,
                                      sub = ix.30,
                                      In = in.est,
                                      upper = c(1, .02, .1),
                                      lower = c(.04, .0001, 0),
                                      cost = "14C + cStock")
names(mod.sens.fits.2ps4r.30) <- names(pars.i.2ps)[ix.30]
save(mod.sens.fits.2ps4r.30, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps4r.30", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps4r.30 <- lapply(mod.sens.fits.2ps4r.30, function(x) x[[1]])
```

```{r report-par-fit}
# load initial parameters as needed
if (!exists("pars.i.2pp")) {
 load("../data/derived/modFit_pars/pars.i.2pp_2020-11-16.Rdata") 
}
if (!exists("pars.i.2ps")) {
  load("../data/derived/modFit_pars/pars.i.2ps_2020-11-16.Rdata")  
}

# load fits as needed
if (!exists("mod.fits.2pp")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2pp_2020-11-16.RData")
}
if (!exists("mod.fits.2pp.p3.5.95")) {
  load("../data/derived/modFit_pars/mod.fits.2pp.p3.5.95_2020-11-16.Rdata")  
}
if (!exists("mod.fits.2ps")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2ps_2020-11-16.Rdata")
}
if (!exists("mod.fits.2pp2")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2pp.flx.stock_2020-12-02.RData")
}
if (!exists("mod.fits.2ps2")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2ps.flx.stock_2020-12-02.Rdata")
}
if (!exists("mod.fits.2pp3")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2pp3_2020-12-08.RData")
}
if (!exists("mod.fits.2ps3")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2ps3_2020-12-08.Rdata")
}
if (!exists("mod.fits.2pp3s")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2pp3s_2020-12-08.RData")
}
if (!exists("mod.fits.2ps3s")) {
 load(file = "../data/derived/modFit_pars/mod.fits.2ps3s_2020-12-08.Rdata")
}

## Par estimates
# 2pp
pars.fit.2pp <- lapply(mod.fits.2pp, "[[", 1)
names(pars.fit.2pp) <- names(pars.i.2pp)[ix.10]
# 2pp gam = [.5, .95]
pars.fit.2pp.p3.5.95 <- lapply(mod.fits.2pp.p3.5.95, "[[", 1)
names(pars.fit.2pp.p3.5.95) <- names(pars.i.2pp)[ix.10]

# 2ps
pars.fit.2ps <- lapply(mod.fits.2ps, "[[", 1)
names(pars.fit.2ps) <- names(pars.i.2ps)[ix.10]


# 2pp2 (input/stock and 14C constraints)
pars.fit.2pp2 <- lapply(mod.fits.2pp2, "[[", 1)
names(pars.fit.2pp2) <- names(pars.i.2pp)[ix.10]
# 2ps2 (input/stock and 14C constraints)
pars.fit.2ps2 <- lapply(mod.fits.2ps2, "[[", 1)
names(pars.fit.2ps2) <- names(pars.i.2ps)[ix.10]

# 2pp3 (14C constraints, constrained par ranges, stock-fit inputs)
pars.fit.2pp3 <- lapply(mod.fits.2pp3, "[[", 1)
names(pars.fit.2pp3) <- names(pars.i.2pp)[ix.10]
# 2ps3 (14C constraints, constrained par ranges, stock-fit inputs)
pars.fit.2ps3 <- lapply(mod.fits.2ps3, "[[", 1)
names(pars.fit.2ps3) <- names(pars.i.2ps)[ix.10]

# 2pp3s (14C constraints, constrained par ranges, stock-fit inputs, + stock constraint)
pars.fit.2pp3s <- lapply(mod.fits.2pp3s, "[[", 1)
names(pars.fit.2pp3s) <- names(pars.i.2pp)[ix.10]
# 2ps3s (14C constraints, constrained par ranges, stock-fit inputs, + stock constraint)
pars.fit.2ps3s <- lapply(mod.fits.2ps3s, "[[", 1)
names(pars.fit.2ps3s) <- names(pars.i.2ps)[ix.10]

## stock & bulk 14C costs only
# 2pp
pars.fit.2pp4.10 <- lapply(mod.fits.2pp4.10, "[[", 1)
names(pars.fit.2pp4.10) <- names(pars.i.2pp)[ix.10]
pars.fit.2pp4.30 <- lapply(mod.fits.2pp4.30, "[[", 1)
names(pars.fit.2pp4.30) <- names(pars.i.2pp)[ix.30]
# 2ps
pars.fit.2ps4.10 <- lapply(mod.fits.2ps4.10, "[[", 1)
names(pars.fit.2ps4.10) <- names(pars.i.2ps)[ix.10]
pars.fit.2ps4.30 <- lapply(mod.fits.2ps4.30, "[[", 1)
names(pars.fit.2ps4.30) <- names(pars.i.2ps)[ix.30]

## stock, bulk and respiration 14C costs
# 2pp
pars.fit.2pp4r.10 <- lapply(mod.fits.2pp4r.10, "[[", 1)
names(pars.fit.2pp4r.10) <- names(pars.i.2pp)[ix.10]
pars.fit.2pp4r.30 <- lapply(mod.fits.2pp4r.30, "[[", 1)
names(pars.fit.2pp4r.30) <- names(pars.i.2pp)[ix.30]
# 2ps
pars.fit.2ps4r.10 <- lapply(mod.fits.2ps4r.10, "[[", 1)
names(pars.fit.2ps4r.10) <- names(pars.i.2ps)[ix.10]
pars.fit.2ps4r.30 <- lapply(mod.fits.2ps4r.30, "[[", 1)
names(pars.fit.2ps4r.30) <- names(pars.i.2ps)[ix.30]

## Summary of fits
# 2pp
pars.fit.2pp.sum <- lapply(mod.fits.2pp, function(x) {
  tryCatch(summary(x), 
           error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(pars.fit.2pp.sum) <- names(pars.fit.2pp)
# 2ps
pars.fit.2ps.sum <- lapply(mod.fits.2ps, function(x) {
  tryCatch(summary(x), 
           error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(pars.fit.2ps.sum) <- names(pars.fit.2ps)

## Summary of errors
# best par set (ssr)
ssr.2pp.df <- data.frame(bind_rows(lapply(mod.fits.2pp, "[", "ssr"), .id = "PMeco_depth"))
ssr.2ps.df <- data.frame(bind_rows(lapply(mod.fits.2ps, "[", "ssr"), .id = "PMeco_depth"))
ssr.2pp2.df <- data.frame(bind_rows(lapply(mod.fits.2pp2, "[", "ssr"), .id = "PMeco_depth"))
ssr.2ps2.df <- data.frame(bind_rows(lapply(mod.fits.2ps2, "[", "ssr"), .id = "PMeco_depth"))
# stock and bulk 14C costs only
ssr.2pp4.10.df <- data.frame(bind_rows(lapply(mod.fits.2pp4.10, "[", "ssr"), .id = "PMeco_depth"))
ssr.2pp4.30.df <- data.frame(bind_rows(lapply(mod.fits.2pp4.30, "[", "ssr"), .id = "PMeco_depth"))
ssr.2ps4.10.df <- data.frame(bind_rows(lapply(mod.fits.2ps4.10, "[", "ssr"), .id = "PMeco_depth"))
ssr.2ps4.30.df <- data.frame(bind_rows(lapply(mod.fits.2ps4.30, "[", "ssr"), .id = "PMeco_depth"))
# stock, bulk and resp 14C costs
ssr.2pp4r.10.df <- data.frame(bind_rows(lapply(mod.fits.2pp4r.10, "[", "ssr"), .id = "PMeco_depth"))
ssr.2pp4r.30.df <- data.frame(bind_rows(lapply(mod.fits.2pp4r.30, "[", "ssr"), .id = "PMeco_depth"))
ssr.2ps4r.10.df <- data.frame(bind_rows(lapply(mod.fits.2ps4r.10, "[", "ssr"), .id = "PMeco_depth"))
ssr.2ps4r.30.df <- data.frame(bind_rows(lapply(mod.fits.2ps4r.30, "[", "ssr"), .id = "PMeco_depth"))

# mean residuals, by var (var_ms)
var_ms.df.fx <- function(mod.fits.ls, costs) {
  df <- data.frame(bind_rows(lapply(mod.fits.ls, "[", "var_ms"), .id = "PMeco_depth"))
  if (length(costs) == 2 ) {
    df$var <- rep(c("resp", "bulkC"), nrow(df)/2)
  } else {
    df$var <- rep(c("resp", "bulkC", "flx.stock"), nrow(df)/3)
  }
  df$var_ms <- round(df$var_ms, 5)
  return(df)
}
var_ms.2pp.df <- var_ms.df.fx(mod.fits.2pp, c("resp", "bulkC"))
var_ms.2pp.p3.5.95.df <- var_ms.df.fx(mod.fits.2pp.p3.5.95, c("resp", "bulkC"))
var_ms.2ps.df <- var_ms.df.fx(mod.fits.2ps, c("resp", "bulkC"))
var_ms.2pp2.df <- var_ms.df.fx(mod.fits.2pp2, c("resp", "bulkC", "flx.stock"))
var_ms.2ps2.df <- var_ms.df.fx(mod.fits.2ps2, c("resp", "bulkC", "flx.stock"))

# bind fitted pars with initial pars into data frame for plotting/summarizing
par.fit.df.fx <- function(mod, pars.fit, pars.i) {
  df <- bind_rows(
    lapply(
      mapply(rbind, 
             pars.fit,
             pars.i,
             SIMPLIFY = FALSE), 
      function(df) {
        df <- data.frame(df)
        if (mod == "2pp") {
          colnames(df) <- c("kfast", "kslow", "gam")
        } else {
          colnames(df) <- c("kfast", "kslow", "a21")
        }
        df$est <- c("fit", "init")
        return(df)
      })
  )
  df$PMeco_depth <- rep(names(pars.i), each = 2)
  df$PM <- substr(df$PMeco_depth, start = 1, stop = 2)
  df$eco <- substr(df$PMeco_depth, start = 3, stop = 4)
  df$depth <- substr(df$PMeco_depth, start = 6, stop = length(df$PMeco_depth))
  return(df)
}


## 2pp
# gam range = [0, 1]
pars.fit.2pp.df <- par.fit.df.fx(mod = "2pp",
                                 pars.fit = pars.fit.2pp,
                                 pars.i = pars.i.2pp[ix.10])
# gam range = [.5, .95]
pars.fit.2pp.p3.5.95.df <- par.fit.df.fx(mod = "2pp",
                                         pars.fit = pars.fit.2pp.p3.5.95,
                                         pars.i = pars.i.2pp[ix.10])
# w/ input/stock cost and gam range = [.5, .95]
pars.fit.2pp2.df <- par.fit.df.fx(mod = "2pp",
                                  pars.fit = pars.fit.2pp2,
                                  pars.i = pars.i.2pp[ix.10])

## 2ps
# a21 range = [0, 1]
pars.fit.2ps.df <- par.fit.df.fx(mod = "2ps",
                                 pars.fit = pars.fit.2ps,
                                 pars.i = pars.i.2ps[ix.10])
# w/ input/stock cost and a21 range = [0, 1]
pars.fit.2ps2.df <- par.fit.df.fx(mod = "2ps",
                                 pars.fit = pars.fit.2ps2,
                                 pars.i = pars.i.2ps[ix.10])

## Constrained par ranges, with and without stock constraint
# w/o stock
pars.fit.2pp3.df <- par.fit.df.fx(mod = "2pp",
                                  pars.fit = pars.fit.2pp3,
                                  pars.i = pars.i.2pp[ix.10])
pars.fit.2ps3.df <- par.fit.df.fx(mod = "2ps",
                                  pars.fit = pars.fit.2ps3,
                                  pars.i = pars.i.2ps[ix.10])
# w/ stock
pars.fit.2pp3s.df <- par.fit.df.fx(mod = "2pp",
                                   pars.fit = pars.fit.2pp3s,
                                   pars.i = pars.i.2ps[ix.10])
pars.fit.2ps3s.df <- par.fit.df.fx(mod = "2ps",
                                   pars.fit = pars.fit.2ps3s,
                                   pars.i = pars.i.2ps[ix.10])
# w/ stock & bulk 14C only
pars.fit.2pp4.10.df <- par.fit.df.fx(mod = "2pp",
                                     pars.fit = pars.fit.2pp4.10,
                                     pars.i = pars.i.2pp[ix.10])
pars.fit.2pp4.30.df <- par.fit.df.fx(mod = "2pp",
                                     pars.fit = pars.fit.2pp4.30,
                                     pars.i = pars.i.2pp[ix.30])
pars.fit.2ps4.10.df <- par.fit.df.fx(mod = "2ps",
                                     pars.fit = pars.fit.2ps4.10,
                                     pars.i = pars.i.2ps[ix.10])
pars.fit.2ps4.30.df <- par.fit.df.fx(mod = "2ps",
                                     pars.fit = pars.fit.2ps4.30,
                                     pars.i = pars.i.2ps[ix.30])

# w/ stock, bulk + resp 14C
pars.fit.2pp4r.10.df <- par.fit.df.fx(mod = "2pp",
                                     pars.fit = pars.fit.2pp4r.10,
                                     pars.i = pars.i.2pp[ix.10])
pars.fit.2pp4r.30.df <- par.fit.df.fx(mod = "2pp",
                                     pars.fit = pars.fit.2pp4r.30,
                                     pars.i = pars.i.2pp[ix.30])
pars.fit.2ps4r.10.df <- par.fit.df.fx(mod = "2ps",
                                     pars.fit = pars.fit.2ps4r.10,
                                     pars.i = pars.i.2ps[ix.10])
pars.fit.2ps4r.30.df <- par.fit.df.fx(mod = "2ps",
                                     pars.fit = pars.fit.2ps4r.30,
                                     pars.i = pars.i.2ps[ix.30])


## Summarize by PM, depth
# 2pp
# PM/depth
pars.fit.2pp.df.PM <- pars.fit.2pp.df %>%
    filter(est == "fit") %>%
    select(!c(est, PMeco_depth, eco)) %>%
    group_by(PM, depth) %>%
    summarize_all(list(mean = mean, sd = sd)) %>%
    mutate_if(is.numeric, format, digits = 2)
# print table
knitr::kable(pars.fit.2pp.df.PM,
             caption = "Mean parameter estimates by parent material (PM)",
             align = "c")
# eco/depth
pars.fit.2pp.df.eco <- pars.fit.2pp.df %>%
  filter(est == "fit") %>%
  mutate(eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
  select(!c(est, PMeco_depth, PM)) %>%
  group_by(eco, depth) %>%
  summarize_all(list(mean = mean, sd = sd)) %>%
  mutate_if(is.numeric, format, digits = 2)
# print table
knitr::kable(pars.fit.2pp.df.eco,
             caption = "Mean parameter estimates by ecosystem (eco)",
             align = "c")
```

```{r sens-fun-fits}
## look at sensFun output
# without constraints
sens.2pp <- lapply(mod.sens.fits.2pp, function(x) x[[2]])
sens.2ps <- lapply(mod.sens.fits.2ps, function(x) x[[2]])
# without stock constraint
sens.2pp3 <- lapply(mod.sens.fits.2pp3, function(x) x[[2]])
sens.2ps3 <- lapply(mod.sens.fits.2ps3, function(x) x[[2]])
# with stock constraint
sens.2pp3s <- lapply(mod.sens.fits.2pp3s, function(x) x[[2]])
sens.2ps3s <- lapply(mod.sens.fits.2ps3s, function(x) x[[2]])
# with stock constraint, w/o resp
sens.2pp4.10 <- lapply(mod.sens.fits.2pp4.10, function(x) x[[2]])
sens.2pp4.30 <- lapply(mod.sens.fits.2pp4.30, function(x) x[[2]])
sens.2ps4.10 <- lapply(mod.sens.fits.2ps4.10, function(x) x[[2]])
sens.2ps4.30 <- lapply(mod.sens.fits.2ps4.30, function(x) x[[2]])
# with stock constraint + resp
sens.2pp4r.10 <- lapply(mod.sens.fits.2pp4r.10, function(x) x[[2]])
sens.2pp4r.30 <- lapply(mod.sens.fits.2pp4r.30, function(x) x[[2]])
sens.2ps4r.10 <- lapply(mod.sens.fits.2ps4r.10, function(x) x[[2]])
sens.2ps4r.30 <- lapply(mod.sens.fits.2ps4r.30, function(x) x[[2]])


# plot sensitivity
# w/o constraints
lapply(sens.2pp, function(x) plot(x, which = c("bulkC", "resp")))
lapply(sens.2ps, function(x) plot(x, which = c("bulkC", "resp")))
# w/o stock constraint
lapply(sens.2pp3, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps3, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2pp3s, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps3s, function(x) plot(x, which = c("bulkC", "cStock")))
# with stock constraint, w/o resp
lapply(sens.2pp4.10, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2pp4.30, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps4.10, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps4.30, function(x) plot(x, which = c("bulkC", "cStock")))
# with stock constraint + resp
lapply(sens.2pp4r.10, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2pp4r.30, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps4r.10, function(x) plot(x, which = c("bulkC", "cStock")))
lapply(sens.2ps4r.30, function(x) plot(x, which = c("bulkC", "cStock")))


# look at identifiability
inden.df.fx <- function(ls, mod) {
  lapply(ls, function(x) {
    df <- collin(x)
    if (mod == "2pp") {
      df$ParCombo <- factor(c("k1 + k2", "k1 + gam", "k2 + gam", "k1 + k2 + gam"))
    } else {
      df$ParCombo <- factor(c("k1 + k2", "k1 + a21", "k2 + a21", "k1 + k2 + a21"))
    }
    return(df)
  })
}

iden.2pp <- inden.df.fx(sens.2pp, mod = "2pp")
iden.2ps <- inden.df.fx(sens.2ps, mod = "2ps")
iden.2pp3 <- inden.df.fx(sens.2pp3, mod = "2pp")
iden.2ps3 <- inden.df.fx(sens.2ps3, mod = "2ps")
iden.2pp3s <- inden.df.fx(sens.2pp3s, mod = "2pp")
iden.2ps3s <- inden.df.fx(sens.2ps3s, mod = "2ps")
# with stock constraint, w/o resp
iden.2pp4.10 <- inden.df.fx(sens.2pp4.10, mod = "2pp")
iden.2pp4.30 <- inden.df.fx(sens.2pp4.30, mod = "2pp")
iden.2ps4.10 <- inden.df.fx(sens.2ps4.10, mod = "2ps")
iden.2ps4.30 <- inden.df.fx(sens.2ps4.30, mod = "2ps")
# with stock constraint + resp
iden.2pp4r.10 <- inden.df.fx(sens.2pp4r.10, mod = "2pp")
iden.2pp4r.30 <- inden.df.fx(sens.2pp4r.30, mod = "2pp")
iden.2ps4r.10 <- inden.df.fx(sens.2ps4r.10, mod = "2ps")
iden.2ps4r.30 <- inden.df.fx(sens.2ps4r.30, mod = "2ps")

# identifiability plot function
coll.plot.fx <- function(df, mod, PMeco_depth, col.max) {
  ggplot(df, aes(N, log(collinearity), color = ParCombo)) +
    geom_hline(yintercept = log(20)) +
    geom_point(size = 3.5, position = position_dodge(width = .1)) +
    scale_y_continuous(limits = c(0, log(col.max))) +
    scale_x_continuous(limits = c(1.5, 3.5), breaks = c(2, 3)) +
    labs(title = paste(PMeco_depth, mod)) +
    theme_bw() +
    theme(panel.grid = element_blank()) +
    if (mod == "2pp" | mod == "2pp + stock") {
     scale_color_manual(
       name = "Parameter combination",
       values = c("k1 + k2" = "#EF476F",
                  "k1 + gam" = "#FFD166",
                  "k2 + gam" = "#118AB2",
                  "k1 + k2 + gam" = "073B4C")) 
    } else {
      scale_color_manual(
        name = "Parameter combination",
        values = c("k1 + k2" = "#EF476F",
                  "k1 + a21" = "#FFD166",
                  "k2 + a21" = "#118AB2",
                  "k1 + k2 + a21" = "073B4C"))
    }
}
lapply(seq_along(iden.2pp), function(i) {
  coll.plot.fx(iden.2pp[[i]], mod = "2pp", names(iden.2pp)[i], max(iden.2pp[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps), function(i) {
  coll.plot.fx(iden.2ps[[i]], mod = "2ps", names(iden.2ps)[i], max(iden.2ps[[i]]["collinearity"]))
})
lapply(seq_along(iden.2pp3), function(i) {
  coll.plot.fx(iden.2pp3[[i]], mod = "2pp", names(iden.2pp3)[i])
})
lapply(seq_along(iden.2pp3s), function(i) {
  coll.plot.fx(iden.2pp3s[[i]], mod = "2pp + stock", names(iden.2pp3s)[i])
})
lapply(seq_along(iden.2ps3), function(i) {
  coll.plot.fx(iden.2ps3[[i]], mod = "2ps", names(iden.2ps3)[i])
})
lapply(seq_along(iden.2ps3s), function(i) {
  coll.plot.fx(iden.2ps3s[[i]], mod = "2ps + stock", names(iden.2ps3s)[i])
})
# stock constraint, w/o resp
col.max <- max(unlist(list(lapply(iden.2ps4.10, function(df) df[["collinearity"]]),
                           lapply(iden.2ps4.30, function(df) df[["collinearity"]]),
                           lapply(iden.2pp4.10, function(df) df[["collinearity"]]),
                           lapply(iden.2pp4.30, function(df) df[["collinearity"]]))))
lapply(seq_along(iden.2ps4.10), function(i) {
  coll.plot.fx(iden.2ps4.10[[i]], mod = "2ps + stock", names(iden.2ps4.10)[i], col.max)
})
lapply(seq_along(iden.2ps4.30), function(i) {
  coll.plot.fx(iden.2ps4.30[[i]], mod = "2ps + stock", names(iden.2ps4.30)[i], col.max)
})
lapply(seq_along(iden.2pp4.10), function(i) {
  coll.plot.fx(iden.2pp4.10[[i]], mod = "2pp + stock", names(iden.2pp4.10)[i], col.max)
})
lapply(seq_along(iden.2pp4.30), function(i) {
  coll.plot.fx(iden.2pp4.30[[i]], mod = "2pp + stock", names(iden.2pp4.30)[i], col.max)
})

# stock constraint + resp
col.max.r <- max(unlist(list(lapply(iden.2ps4r.10, function(df) df[["collinearity"]]),
                             lapply(iden.2ps4r.30, function(df) df[["collinearity"]]),
                             lapply(iden.2pp4r.10, function(df) df[["collinearity"]]),
                             lapply(iden.2pp4r.30, function(df) df[["collinearity"]]))))
lapply(seq_along(iden.2pp4r.10), function(i) {
  coll.plot.fx(iden.2pp4r.10[[i]], mod = "2pp", names(iden.2pp4r.10)[i], col.max)
})
lapply(seq_along(iden.2pp4r.30), function(i) {
  coll.plot.fx(iden.2pp4r.30[[i]], mod = "2ps", names(iden.2pp4r.30)[i], col.max)
})
lapply(seq_along(iden.2ps4r.10), function(i) {
  coll.plot.fx(iden.2ps4r.10[[i]], mod = "2ps + stock", names(iden.2ps4r.10)[i], col.max)
})
lapply(seq_along(iden.2ps4r.30), function(i) {
  coll.plot.fx(iden.2ps4r.30[[i]], mod = "2ps + stock", names(iden.2ps4r.30)[i], col.max)
})
```


```{r plot-modFit-pars}
## plot pars
par.plot.fx <- function(mod, depth, par.df, initial = FALSE) {
  par.df %>%
    { if (initial == TRUE) . else filter(., est == "fit") } %>%
    filter(depth == depth) %>%
    pivot_longer(!(est:depth), names_to = "par", values_to = "value") %>%
    mutate(PM = factor(PM),
           eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
    ggplot(., aes(par, value, color = PM, shape = eco)) +
    # geom_jitter(size = 4) +
    geom_point(size = 4, position = position_dodge(width = .5)) +
    scale_color_manual(name = "parent material",
                      labels = c("AN" = "andesite",
                                 "BS" = "basalt",
                                 "GR" = "granite"),
                      values = c("AN" = "blue", 
                                 "BS" = "red", 
                                 "GR" = "darkgray")) +
    facet_wrap(. ~ par, scales = "free") +
    ggtitle(paste0("modFit pars ", mod, " ", depth)) +
    theme_bw() +
    theme(panel.grid.minor = element_blank())
}
# 0-10
# 2pp
par.plot.fx(mod = "2pp",
            depth = "0-10",
            par.df = pars.fit.2pp.df,
            initial = FALSE)
# 2pp, gam = [.5,.95]
par.plot.fx(mod = "2pp (gam = [0.5, 0.95])",
            depth = "0-10",
            par.df = pars.fit.2pp.p3.5.95.df,
            initial = FALSE)
# 2pp2
par.plot.fx(mod = "2pp2",
            depth = "0-10",
            par.df = pars.fit.2pp2.df,
            initial = FALSE)
# 2ps
par.plot.fx(mod = "2ps",
            depth = "0-10",
            par.df = pars.fit.2ps.df,
            initial = FALSE)
# 2ps2
par.plot.fx(mod = "2ps2",
            depth = "0-10",
            par.df = pars.fit.2ps2.df,
            initial = FALSE)

# w/ and w/o stock constraint
par.plot.fx(mod = "2pp3",
            depth = "0-10",
            par.df = pars.fit.2pp3.df,
            initial = FALSE)
par.plot.fx(mod = "2pp3s",
            depth = "0-10",
            par.df = pars.fit.2pp3s.df,
            initial = FALSE)
par.plot.fx(mod = "2ps3",
            depth = "0-10",
            par.df = pars.fit.2ps3.df,
            initial = FALSE)
par.plot.fx(mod = "2ps3s",
            depth = "0-10",
            par.df = pars.fit.2ps3s.df,
            initial = FALSE)

## flux est inputs by eco
# stock and bulk 14C only
par.plot.fx(mod = "2pp4",
            depth = "0-10",
            par.df = pars.fit.2pp4.10.df,
            initial = FALSE)
par.plot.fx(mod = "2pp4",
            depth = "20-30",
            par.df = pars.fit.2pp4.30.df,
            initial = FALSE)
par.plot.fx(mod = "2ps4",
            depth = "0-10",
            par.df = pars.fit.2ps4.10.df,
            initial = FALSE)
par.plot.fx(mod = "2ps4",
            depth = "20-30",
            par.df = pars.fit.2ps4.30.df,
            initial = FALSE)

# stock and bulk + resp 14C
par.plot.fx(mod = "2pp4r",
            depth = "0-10",
            par.df = pars.fit.2pp4r.10.df,
            initial = FALSE)
par.plot.fx(mod = "2pp4r",
            depth = "20-30",
            par.df = pars.fit.2pp4r.30.df,
            initial = FALSE)
par.plot.fx(mod = "2ps4r",
            depth = "0-10",
            par.df = pars.fit.2ps4r.10.df,
            initial = FALSE)
par.plot.fx(mod = "2ps4r",
            depth = "20-30",
            par.df = pars.fit.2ps4r.30.df,
            initial = FALSE)
```

```{r fit-soc-in}
## Find best inputs
# 2pp
in.fit.2pp <- lapply(seq_along(pars.fit.2pp), function(i) {
  PMeco_depth <- names(pars.fit.2pp)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.fit.2pp[[i]], in.i[ix.10][[i]], SOC))
})
names(in.fit.2pp) <- names(mod.fits.2pp)
# 2pp gam = [.5, .95]
in.fit.2pp.p3.5.95 <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  PMeco_depth <- names(pars.fit.2pp.p3.5.95)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.fit.2pp.p3.5.95[[i]], in.i[ix.10][[i]], SOC))
})
names(in.fit.2pp.p3.5.95) <- names(mod.fits.2pp.p3.5.95)
# 2pp2
in.fit.2pp2 <- lapply(seq_along(pars.fit.2pp2), function(i) {
  PMeco_depth <- names(pars.fit.2pp2)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2pp", pars.fit.2pp2[[i]], in.flx.stock[[i]], SOC))
})
names(in.fit.2pp2) <- names(mod.fits.2pp2)
# 2ps
in.fit.2ps <- lapply(seq_along(pars.fit.2ps), function(i) {
  PMeco_depth <- names(pars.fit.2ps)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2ps", pars.fit.2ps[[i]], in.i[ix.10][[i]], SOC))
})
names(in.fit.2ps) <- names(mod.fits.2ps)
# 2ps2
in.fit.2ps2 <- lapply(seq_along(pars.fit.2ps2), function(i) {
  PMeco_depth <- names(pars.fit.2ps2)[i]
  SOC <- csoc.19.0_30[[PMeco_depth]][ ,"lyr_soc"]
  return(in.fit.fx("2ps", pars.fit.2ps2[[i]], in.flx.stock[[i]], SOC))
})
names(in.fit.2ps2) <- names(mod.fits.2ps2)

## Calc modeled stocks and compare with measured stocks
# 2pp
mod.socs.2pp.ls <- lapply(seq_along(pars.fit.2pp), function(i) {
  soc.fx("2pp", pars.fit.2pp[[i]], in.fit.2pp[[i]])
})
names(mod.socs.2pp.ls) <- names(pars.fit.2pp)
socs.2pp.ls <- mapply(cbind,
                      csoc.19.0_30[ix.10], 
                      lapply(mod.socs.2pp.ls, colSums), 
                      SIMPLIFY = FALSE)
# 2pp gam = [.5, .95]
mod.socs.2pp.p3.5.95.ls <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  soc.fx("2pp", pars.fit.2pp.p3.5.95[[i]], in.fit.2pp.p3.5.95[[i]])
})
names(mod.socs.2pp.p3.5.95.ls) <- names(pars.fit.2pp.p3.5.95)
socs.2pp.p3.5.95ls <- mapply(cbind,
                             csoc.19.0_30[ix.10], 
                             lapply(mod.socs.2pp.p3.5.95.ls, colSums), 
                             SIMPLIFY = FALSE)
# 2pp2
mod.socs.2pp2.ls <- lapply(seq_along(pars.fit.2pp2), function(i) {
  soc.fx("2pp", pars.fit.2pp2[[i]], in.fit.2pp2[[i]])
})
names(mod.socs.2pp2.ls) <- names(pars.fit.2pp2)

# 2ps
mod.socs.2ps.ls <- lapply(seq_along(pars.fit.2ps), function(i) {
  soc.fx("2ps", pars.fit.2ps[[i]], in.fit.2ps[[i]])
})
names(mod.socs.2ps.ls) <- names(pars.fit.2ps)
socs.2ps.ls <- mapply(cbind,
                      csoc.19.0_30[ix.10], 
                      lapply(mod.socs.2ps.ls, colSums), 
                      SIMPLIFY = FALSE)
# 2ps2
mod.socs.2ps2.ls <- lapply(seq_along(pars.fit.2ps2), function(i) {
  soc.fx("2ps", pars.fit.2ps2[[i]], in.fit.2ps2[[i]])
})
names(mod.socs.2ps2.ls) <- names(pars.fit.2ps2)

## stock and bulk 14C costs only
# 2pp
mod.socs.2pp4.10.ls <- lapply(seq_along(pars.fit.2pp4.10), function(i) {
  soc.fx("2pp", pars.fit.2pp4.10[[i]], in.est[ix.10][[i]])
})
names(mod.socs.2pp4.10.ls) <- names(pars.fit.2pp4.10)
mod.socs.2pp4.30.ls <- lapply(seq_along(pars.fit.2pp4.30), function(i) {
  soc.fx("2pp", pars.fit.2pp4.30[[i]], in.est[ix.30][[i]])
})
names(mod.socs.2pp4.30.ls) <- names(pars.fit.2pp4.30)
# 2ps
mod.socs.2ps4.10.ls <- lapply(seq_along(pars.fit.2ps4.10), function(i) {
  soc.fx("2ps", pars.fit.2ps4.10[[i]], in.est[ix.10][[i]])
})
names(mod.socs.2ps4.10.ls) <- names(pars.fit.2ps4.10)
mod.socs.2ps4.30.ls <- lapply(seq_along(pars.fit.2ps4.30), function(i) {
  soc.fx("2ps", pars.fit.2ps4.30[[i]], in.est[ix.30][[i]])
})
names(mod.socs.2ps4.30.ls) <- names(pars.fit.2ps4.30)

## stock and bulk + resp 14C costs
# 2pp
mod.socs.2pp4r.10.ls <- lapply(seq_along(pars.fit.2pp4r.10), function(i) {
  soc.fx("2pp", pars.fit.2pp4r.10[[i]], in.est[ix.10][[i]])
})
names(mod.socs.2pp4r.10.ls) <- names(pars.fit.2pp4r.10)
mod.socs.2pp4r.30.ls <- lapply(seq_along(pars.fit.2pp4r.30), function(i) {
  soc.fx("2pp", pars.fit.2pp4r.30[[i]], in.est[ix.30][[i]])
})
names(mod.socs.2pp4r.30.ls) <- names(pars.fit.2pp4r.30)
# 2ps
mod.socs.2ps4r.10.ls <- lapply(seq_along(pars.fit.2ps4r.10), function(i) {
  soc.fx("2ps", pars.fit.2ps4r.10[[i]], in.est[ix.10][[i]])
})
names(mod.socs.2ps4r.10.ls) <- names(pars.fit.2ps4r.10)
mod.socs.2ps4r.30.ls <- lapply(seq_along(pars.fit.2ps4r.30), function(i) {
  soc.fx("2ps", pars.fit.2ps4r.30[[i]], in.est[ix.30][[i]])
})
names(mod.socs.2ps4r.30.ls) <- names(pars.fit.2ps4r.30)


## Return data frames of model fits with adjusted inputs and optimal parameters
# 2pp
Twopp.fits <- lapply(seq_along(pars.fit.2pp), function(i) {
  par.fx(pars.fit.2pp[[i]], in.fit.2pp[[i]], verbose = FALSE, mod = "2pp")
})
names(Twopp.fits) <- names(pars.fit.2pp)
# 2pp gam = [.5, .95]
Twopp.p3.5.95.fits <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  par.fx(pars.fit.2pp.p3.5.95[[i]], in.fit.2pp.p3.5.95[[i]], verbose = FALSE, mod = "2pp")
})
names(Twopp.p3.5.95.fits) <- names(pars.fit.2pp.p3.5.95)
# 2pp2
Twopp2.fits <- lapply(seq_along(pars.fit.2pp2), function(i) {
  par.fx(pars.fit.2pp2[[i]], in.fit.2pp2[[i]], verbose = FALSE, mod = "2pp")
})
names(Twopp2.fits) <- names(pars.fit.2pp2)
# 2ps
Twops.fits <- lapply(seq_along(pars.fit.2ps), function(i) {
  par.fx(pars.fit.2ps[[i]], in.fit.2ps[[i]], verbose = FALSE, mod = "2ps")
})
names(Twops.fits) <- names(pars.fit.2ps)
# 2ps2
Twops2.fits <- lapply(seq_along(pars.fit.2ps2), function(i) {
  par.fx(pars.fit.2ps2[[i]], in.fit.2ps2[[i]], verbose = FALSE, mod = "2ps", pass = TRUE)
})
names(Twops2.fits) <- names(pars.fit.2ps2)

## stock and bulk 14C costs only
# 2pp
Twopp4.10.fits <- lapply(seq_along(pars.fit.2pp4.10), function(i) {
  par.fx(pars.fit.2pp4.10[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2pp", pass = FALSE)
})
names(Twopp4.10.fits) <- names(pars.fit.2pp4.10)
Twopp4.30.fits <- lapply(seq_along(pars.fit.2pp4.30), function(i) {
  par.fx(pars.fit.2pp4.30[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2pp", pass = FALSE)
})
names(Twopp4.30.fits) <- names(pars.fit.2pp4.30)
# 2ps
Twops4.10.fits <- lapply(seq_along(pars.fit.2ps4.10), function(i) {
  par.fx(pars.fit.2ps4.10[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE)
})
names(Twops4.10.fits) <- names(pars.fit.2ps4.10)
Twops4.30.fits <- lapply(seq_along(pars.fit.2ps4.30), function(i) {
  par.fx(pars.fit.2ps4.30[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2ps", pass = FALSE)
})
names(Twops4.30.fits) <- names(pars.fit.2ps4.30)

## stock and bulk + resp 14C costs
# 2pp
Twopp4r.10.fits <- lapply(seq_along(pars.fit.2pp4r.10), function(i) {
  par.fx(pars.fit.2pp4r.10[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2pp", pass = FALSE)
})
names(Twopp4r.10.fits) <- names(pars.fit.2pp4r.10)
Twopp4r.30.fits <- lapply(seq_along(pars.fit.2pp4r.30), function(i) {
  par.fx(pars.fit.2pp4r.30[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2pp", pass = FALSE)
})
names(Twopp4r.30.fits) <- names(pars.fit.2pp4r.30)
# 2ps
Twops4r.10.fits <- lapply(seq_along(pars.fit.2ps4r.10), function(i) {
  par.fx(pars.fit.2ps4r.10[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE)
})
names(Twops4r.10.fits) <- names(pars.fit.2ps4r.10)
Twops4r.30.fits <- lapply(seq_along(pars.fit.2ps4r.30), function(i) {
  par.fx(pars.fit.2ps4r.30[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2ps", pass = FALSE)
})
names(Twops4r.30.fits) <- names(pars.fit.2ps4r.30)
```

```{r plot-soc-stocks}
# Plot optimized model SOC stocks
mod.socs.df.fx <- function(mod, mod.socs.ls, pools) {
  n <- vapply(mod.socs.ls, nrow, numeric(1))
  return(data.frame(SOC = do.call(rbind, mod.socs.ls),
                    pool = rep(pools, length(mod.socs.ls)),
                    PMeco_depth = rep(names(mod.socs.ls), n),
                    Model = rep(mod, sum(n))))       
}
# run fx
# mod.socs.2p.df <- rbind(mod.socs.df.fx("2pp", mod.socs.2pp.ls, c("fast", "slow"))
#                         ,mod.socs.df.fx("2ps", mod.socs.2ps.ls, c("fast", "slow"))
#                         ,mod.socs.df.fx("2pp [.5,.95]", mod.socs.2pp.p3.5.95.ls, c("fast", "slow"))
#                         ,mod.socs.df.fx("2ps2", mod.socs.2ps2.ls, c("fast", "slow"))
#                         ,mod.socs.df.fx("2pp2", mod.socs.2pp2.ls, c("fast", "slow"))
#                         )
mod.socs.2p.df <- rbind(mod.socs.df.fx("2pp", mod.socs.2pp.ls, c("fast", "slow"))
                        ,mod.socs.df.fx("2ps", mod.socs.2ps.ls, c("fast", "slow"))
                        )


# stocks and bulk 14C only
mod.socs.2p4.10.df <- rbind(mod.socs.df.fx("2pp4 0-10", mod.socs.2pp4.10.ls, c("fast", "slow")), mod.socs.df.fx("2ps4 0-10", mod.socs.2ps4.10.ls, c("fast", "slow")))
mod.socs.2p4.30.df <- rbind(mod.socs.df.fx("2pp4 20-30", mod.socs.2pp4.30.ls, c("fast", "slow")) ,mod.socs.df.fx("2ps4 20-30", mod.socs.2ps4.30.ls, c("fast", "slow")))

# stocks and bulk + resp 14C
mod.socs.2p4r.10.df <- rbind(mod.socs.df.fx("2pp4r 0-10", mod.socs.2pp4r.10.ls, c("fast", "slow")), mod.socs.df.fx("2ps4r 0-10", mod.socs.2ps4r.10.ls, c("fast", "slow")))
mod.socs.2p4r.30.df <- rbind(mod.socs.df.fx("2pp4r 20-30", mod.socs.2pp4r.30.ls, c("fast", "slow")) ,mod.socs.df.fx("2ps4r 20-30", mod.socs.2ps4r.30.ls, c("fast", "slow")))

# combine inputs to compare
# in.fits.df <- pivot_longer(do.call(bind_rows, list(in.fit.2pp,
#                                                 in.fit.2pp.p3.5.95,
#                                                 in.fit.2pp2,
#                                                 in.fit.2ps,
#                                                 in.fit.2ps2)),
#                            everything(),
#                            names_to = "PMeco_depth",
#                            values_to = "inputs")
# in.fits.df$mod <- rep(c("2pp",
#                         "2pp.5.95",
#                         "2pp2",
#                         "2ps",
#                         "2ps2"),
#                       each = 9)
in.fits.df <- pivot_longer(do.call(bind_rows, list(in.fit.2pp,
                                                   in.fit.2ps)),
                           everything(),
                           names_to = "PMeco_depth",
                           values_to = "inputs")
in.fits.df$mod <- rep(c("2pp",
                        "2ps"),
                      each = 9)
                        
## plot stocks
# stock and bulk 14C only
mod.socs.2p4.10.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
mod.socs.2p4.30.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# stock and bulk + resp 14C
mod.socs.2p4r.10.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
mod.socs.2p4r.30.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# inputs
in.fits.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         # Model = factor(Model, levels = c("2pp [.5,.95]", "2pp", "2ps")),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(mod, inputs, fill = mod)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```

```{r plot-opt-mod-2pp}
# plot fx
Twop.fit.plot.fx <- function(fit1, fit1.name, fit2, fit2.name, fit3 = NULL, fit3.name = NULL) {
  lapply(seq_along(fit1), function(i) {
    PMeco <- substr(names(fit1)[i], 1, 4)
    lyr_bot <- substr(names(fit1)[i], 
                      nchar(names(fit1)[i]) - 1, 
                      nchar(names(fit1)[i]))
    lyr_top <- ifelse(lyr_bot == 10, 0, ifelse(lyr_bot == 20, 10, 20))
    PMeco_depth <- names(fit1)[i]
    con.df <- con.df.fx(PMeco_depth)
    plot.df <- rbind(fit1[[i]],
                     fit2[[i]],
                     fit3[[i]])
    plot.df$Model <- factor(c(rep(fit1.name, nrow(fit1[[i]])),
                              rep(fit2.name, nrow(fit2[[i]])),
                              rep(fit3.name, nrow(fit3[[i]]))),
                            levels = c(fit1.name, fit2.name, fit3.name))
    return(plot.df %>%
             filter(pool == "bulk C" | pool == "respiration" | pool == "atm") %>%
             ggplot(., aes(years, d14C, color = pool)) +
             geom_path(aes(linetype = Model)) +
             geom_point(data = con.df, aes(Year, d14c, color = pool), size = 3) +
             scale_color_manual(
               name = "Model pool",
               values = c("atm" = 8,
                          "bulk C" = "black",
                          "fast" = "#D81B60",
                          "slow" = "#1E88E5",
                          "respiration" = "#FFC107")) +
             scale_x_continuous(limits = c(1950, 2022)) +
             ggtitle(paste0(PMeco_depth, " 2p mod fits")) +
             xlab("Year") +
             ylab(expression(''*Delta*''^14*'C (‰)')) +
             theme_bw() +
             theme(panel.grid = element_blank()))
  })
}
# 2p modFit optimal model comparison
Twop.fits.plots <- Twop.fit.plot.fx(Twopp.fits, "2pp", Twops.fits, "2ps")
Twop.fits.plots
# Twop.fits.plots2 <- Twop.fit.plot.fx(Twopp.fits, "2pp", Twopp.p3.5.95.fits, "2pp gam = [.5, .95]")
# Twop.fits.plots2
Twop.fits.plots3 <- Twop.fit.plot.fx(Twopp.p3.5.95.fits, "2pp gam = [.5, .95]", Twopp2.fits, "2pp2")
Twop.fits.plots3

## compare fits w/ and w/o resp constraint (2p4 mods)
# 2pp
Twopp4.fits.plots.10 <- Twop.fit.plot.fx(Twopp4.10.fits, "2pp4 0-10cm w/o resp", Twopp4r.10.fits, "2pp4r 0-10cm w/ resp")
Twopp4.fits.plots.30 <- Twop.fit.plot.fx(Twopp4.30.fits, "2pp4 20-30cm w/o resp", Twopp4r.10.fits, "2pp4r 20-30cm w/ resp")
# 2ps
Twops4.fits.plots.10 <- Twop.fit.plot.fx(Twops4.10.fits, "2ps4 0-10cm w/o resp", Twops4r.10.fits, "2ps4r 0-10cm w/ resp")
Twops4.fits.plots.30 <- Twop.fit.plot.fx(Twops4.30.fits, "2ps4 20-30cm w/o resp", Twops4r.10.fits, "2ps4r 20-30cm w/ resp")
# plot
Twopp4.fits.plots.10
Twopp4.fits.plots.30
Twops4.fits.plots.10
Twops4.fits.plots.30
```

```{r SAB-obs}
p <- sra.ts.all %>%
    filter(d14c > -200) %>%
    filter(ECO != "rf") %>%
    filter(lyr_bot == 20) %>%
    filter(year != 2009) %>%
    ggplot(., aes(year, d14c)) +
    geom_path(data = atm.14c) +
    geom_point(aes(color = pm, shape = ecoType), size = 3.5) +
    geom_path(aes(color = pm, linetype = Type), size = 1, alpha = 0.3) +
    geom_errorbar(
        aes(ymin = d14c_l, 
            ymax = d14c_u,
            color = pm), 
        width = .5) +
    scale_color_manual(name = "Parent material",
                       values = c("andesite" = "blue", 
                                  "basalt" = "red", 
                                  "granite" = "darkgray")) +
    scale_shape_manual(name = "Ecosystem (type)",
                       values = c("warm (inc)" = 0,
                                  "cool (inc)" = 1,
                                  "cold (inc)" = 2,
                                  "warm (bulk)" = 15,
                                  "cool (bulk)" = 16,
                                  "cold (bulk)" = 17)) +
    facet_grid(rows = vars(eco), cols = vars(pm)) +
    ylab(expression(Delta*''^14*'C (‰)')) +
    xlab("Year") +
    ggtitle("Bulk/inc 10-20 cm") +
    theme_bw() +
    theme(panel.grid = element_blank(),
          axis.text.x = element_text(size = 8))
ggsave("sra.ts.ppwf20.blk.inc.pdf", p, dpi = 300, width = 6.97, height = 5, units = "in")
# inc/bulk profiles
p <- sra.19.01.09 %>%
  filter(lyr_bot < 31) %>%
  select(Year, PM, ECO, PMeco, lyr_bot, d14c, d14c_sd) %>%
  mutate(Type = "bulk",
         d14c_u = d14c + d14c_sd,
         d14c_l = d14c - d14c_sd,
         year = as.numeric(as.character(Year))) %>%
  select(-d14c_sd) %>%
  bind_rows(.,
            sra.19.01.inc %>%
              select(year, PM, ECO, PMeco, lyr_bot, d14c, d14c_min, d14c_max) %>%
              rename(d14c_l = d14c_min,
                     d14c_u = d14c_max) %>%
              mutate(Type = "inc")
  ) %>%
  mutate(depth = factor(lyr_bot),
         eco = factor(ifelse(ECO == "pp", "warm",
                      ifelse(ECO == "wf", "cool", "cold")),
                      levels = c("warm", "cool", "cold")),
         pm = ifelse(PM == "AN", "andesite",
                     ifelse(PM == "BS", "basalt", "granite")),
         ecoType = paste0(eco, " (", Type, ")"))
ggsave("sra.ts.ppwf20.blk.pdf", p, dpi = 300, width = 6.97, height = 5, units = "in")
```

```{r SAB-modfits}
### Run modfit
## 14C bulk only
# 0-10
mod.sens.fits.2ps.10b <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps, 
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, .02, .999),
                                     lower = c(.02, .0001, .001),
                                     cost = "14C bulk only")
names(mod.sens.fits.2ps.10b) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps.10b, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.10b", "_", Sys.Date(), ".Rdata"))
# # 20-30
# mod.sens.fits.2ps.30b <- mod.fits.fx(mod = "2ps",
#                                      pars = pars.i.2ps, 
#                                      sub = ix.30,
#                                      In = in.est,
#                                      upper = c(1, .02, .15),
#                                      lower = c(.005, .0001, .0004),
#                                      cost = "14C bulk only")
# names(mod.sens.fits.2ps.30b) <- names(pars.i.2ps)[ix.30]
# save(mod.sens.fits.2ps.30b, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30b", "_", Sys.Date(), ".Rdata"))
# mod.fits.2ps.30b <- lapply(mod.sens.fits.2ps.30b, function(x) x[[1]])

## 14C (bulk + resp)
# 0-10
mod.sens.fits.2ps.10br <- mod.fits.fx(mod = "2ps",
                                     pars = pars.i.2ps, 
                                     sub = ix.10,
                                     In = in.est,
                                     upper = c(1, .02, .999),
                                     lower = c(.02, .0001, .001),
                                     cost = "14C")
names(mod.sens.fits.2ps.10br) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps.10br, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.10br", "_", Sys.Date(), ".Rdata"))
# 10-20, lag = 5
mod.sens.fits.2ps.20br.l <- mod.fits.fx(mod = "2ps",
                                       pars = pars.i.2ps,
                                       sub = ix.20,
                                       In = in.est,
                                      lag = 5,
                                       upper = c(1, .02, .99),
                                       lower = c(.02, .0001, .01),
                                       cost = "14C")
names(mod.sens.fits.2ps.20br.l) <- names(pars.i.2ps)[ix.20]
save(mod.sens.fits.2ps.20br.l, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.20br.l", "_", Sys.Date(), ".Rdata"))
# # 20-30
# mod.sens.fits.2ps.30br <- mod.fits.fx(mod = "2ps",
#                                      pars = pars.i.2ps, 
#                                      sub = ix.30,
#                                      In = in.est,
#                                      upper = c(1, .02, .15),
#                                      lower = c(.005, .0001, .0004),
#                                      cost = "14C")
# names(mod.sens.fits.2ps.30br) <- names(pars.i.2ps)[ix.30]
# save(mod.sens.fits.2ps.30br, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30br", "_", Sys.Date(), ".Rdata"))
# mod.fits.2ps.30br <- lapply(mod.sens.fits.2ps.30br, function(x) x[[1]])

## 14C bulk + stocks
# 0-10
mod.sens.fits.2ps.10bs <- mod.fits.fx(mod = "2ps",
                                      pars = pars.i.2ps,
                                      sub = ix.10,
                                      In = in.est,
                                      upper = c(1, .02, .999),
                                      lower = c(.02, .0001, .001),
                                      cost = "14C bulk + cStock")
names(mod.sens.fits.2ps.10bs) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps.10bs, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.10bs", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps.10bs <- lapply(mod.sens.fits.2ps.10bs, function(x) x[[1]])
# # 20-30
# mod.sens.fits.2ps.30b <- mod.fits.fx(mod = "2ps",
#                                      pars = pars.i.2ps, 
#                                      sub = ix.30,
#                                      In = in.est,
#                                      upper = c(1, .02, .15),
#                                      lower = c(.005, .0001, .0004),
#                                      cost = "14C bulk only")
# names(mod.sens.fits.2ps.30b) <- names(pars.i.2ps)[ix.30]
# save(mod.sens.fits.2ps.30b, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30b", "_", Sys.Date(), ".Rdata"))
# mod.fits.2ps.30b <- lapply(mod.sens.fits.2ps.30b, function(x) x[[1]])

## 14C + cStock (14C resp, 14C bulk, stocks)
# 0-10
mod.sens.fits.2ps.10rbs <- mod.fits.fx(mod = "2ps",
                                       pars = pars.i.2ps,
                                       sub = ix.10,
                                       In = in.est,
                                       upper = c(1, .02, .999),
                                       lower = c(.02, .0001, .001),
                                       cost = "14C + cStock")
names(mod.sens.fits.2ps.10rbs) <- names(pars.i.2ps)[ix.10]
save(mod.sens.fits.2ps.10rbs, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.10rbs", "_", Sys.Date(), ".Rdata"))
mod.fits.2ps.10rbs <- lapply(mod.sens.fits.2ps.10rbs, function(x) x[[1]])
# 10-20
# w/o lag
mod.sens.fits.2ps.20rbs <- mod.fits.fx(mod = "2ps",
                                       pars = pars.i.2ps,
                                       sub = ix.20,
                                       In = in.est,
                                       upper = c(1, .02, .99),
                                       lower = c(.02, .0001, .01),
                                       cost = "14C + cStock")
names(mod.sens.fits.2ps.20rbs) <- names(pars.i.2ps)[ix.20]
save(mod.sens.fits.2ps.20rbs, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.20rbs", "_", Sys.Date(), ".Rdata"))
# w/ lag = 12
mod.sens.fits.2ps.20rbs.l <- mod.fits.fx(mod = "2ps",
                                       pars = pars.i.2ps,
                                       sub = ix.20,
                                       In = in.est,
                                       lag = 12,
                                       upper = c(1, .02, .99),
                                       lower = c(.02, .0001, .01),
                                       cost = "14C + cStock")
names(mod.sens.fits.2ps.20rbs.l) <- names(pars.i.2ps)[ix.20]
save(mod.sens.fits.2ps.20rbs.l, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.20rbs.l", "_", Sys.Date(), ".Rdata"))
# # 20-30
# mod.sens.fits.2ps.30b <- mod.fits.fx(mod = "2ps",
#                                      pars = pars.i.2ps, 
#                                      sub = ix.30,
#                                      In = in.est,
#                                      upper = c(1, .02, .15),
#                                      lower = c(.005, .0001, .0004),
#                                      cost = "14C bulk only")
# names(mod.sens.fits.2ps.30b) <- names(pars.i.2ps)[ix.30]
# save(mod.sens.fits.2ps.30b, file = paste0("../data/derived/modFit_pars/", "mod.fits.2ps.30b", "_", Sys.Date(), ".Rdata"))
# mod.fits.2ps.30b <- lapply(mod.sens.fits.2ps.30b, function(x) x[[1]])
```

```{r SAB-mod-fits}
# SAB fits
load("../data/derived/modFit_pars/mod.fits.2ps.10b_2021-04-07.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.10br_2021-04-07.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.10bs_2021-04-07.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.10rbs_2021-04-07.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.20rbs_2021-04-12.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.20rbs.l_2021-04-13.Rdata")
load("../data/derived/modFit_pars/mod.fits.2ps.20br.l_2021-04-13.Rdata")
load("../data/derived/modFit_pars/pars.i.2ps_2021-04-06.Rdata")

# extract mod fits
mod.fits.2ps.10b <- lapply(mod.sens.fits.2ps.10b, function(x) x[[1]])
mod.fits.2ps.10bs <- lapply(mod.sens.fits.2ps.10bs, function(x) x[[1]])
mod.fits.2ps.10br <- lapply(mod.sens.fits.2ps.10br, function(x) x[[1]])
mod.fits.2ps.10rbs <- lapply(mod.sens.fits.2ps.10rbs, function(x) x[[1]])
mod.fits.2ps.20rbs <- lapply(mod.sens.fits.2ps.20rbs, function(x) x[[1]])
mod.fits.2ps.20rbs.l <- lapply(mod.sens.fits.2ps.20rbs.l, function(x) x[[1]])
mod.fits.2ps.20br.l <- lapply(mod.sens.fits.2ps.20br.l, function(x) x[[1]]) 
  
# Sensitivity/Identifiability
#####
# extract at sensFun output
sens.2ps.10b <- lapply(mod.sens.fits.2ps.10b, function(x) x[[2]])
sens.2ps.10br <- lapply(mod.sens.fits.2ps.10br, function(x) x[[2]])
sens.2ps.10bs <- lapply(mod.sens.fits.2ps.10bs, function(x) x[[2]])
sens.2ps.10rbs <- lapply(mod.sens.fits.2ps.10rbs, function(x) x[[2]])
sens.2ps.20rbs <- lapply(mod.sens.fits.2ps.20rbs, function(x) x[[2]])
sens.2ps.20rbs.l <- lapply(mod.sens.fits.2ps.20rbs.l, function(x) x[[2]])
sens.2ps.20br.l <- lapply(mod.sens.fits.2ps.20br.l, function(x) x[[2]])

# plot sensitivity
lapply(sens.2ps.10b, function(x) plot(x, which = c("bulkC", "resp")))
lapply(sens.2ps.10br, function(x) plot(x, which = c("bulkC", "resp")))
lapply(sens.2ps.10bs, function(x) plot(x, which = c("bulkC", "resp")))
lapply(sens.2ps.10rbs, function(x) plot(x, which = c("bulkC", "resp")))

# look at identifiability
iden.2ps.10b <- inden.df.fx(sens.2ps.10b, mod = "2ps")
iden.2ps.10br <- inden.df.fx(sens.2ps.10br, mod = "2ps")
iden.2ps.10bs <- inden.df.fx(sens.2ps.10bs, mod = "2ps")
iden.2ps.10rbs <- inden.df.fx(sens.2ps.10rbs, mod = "2ps")
iden.2ps.20rbs <- inden.df.fx(sens.2ps.20rbs, mod = "2ps")
iden.2ps.20rbs.l <- inden.df.fx(sens.2ps.20rbs.l, mod = "2ps")
iden.2ps.20br.l <- inden.df.fx(sens.2ps.20br.l, mod = "2ps")

# plot
lapply(seq_along(iden.2ps.10bs), function(i) {
  coll.plot.fx(iden.2ps.10bs[[i]], mod = "2ps", 
               names(iden.2ps.10bs)[i], 
               max(iden.2ps.10bs[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.10br), function(i) {
  coll.plot.fx(iden.2ps.10br[[i]], mod = "2ps", 
               names(iden.2ps.10br)[i], 
               max(iden.2ps.10br[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.10rbs), function(i) {
  coll.plot.fx(iden.2ps.10rbs[[i]], mod = "2ps", 
               names(iden.2ps.10rbs)[i], 
               max(iden.2ps.10rbs[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.20rbs), function(i) {
  coll.plot.fx(iden.2ps.20rbs[[i]], mod = "2ps", 
               names(iden.2ps.20rbs)[i], 
               max(iden.2ps.20rbs[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.20rbs.l), function(i) {
  coll.plot.fx(iden.2ps.20rbs.l[[i]], mod = "2ps", 
               names(iden.2ps.20rbs.l)[i], 
               max(iden.2ps.20rbs.l[[i]]["collinearity"]))
})
lapply(seq_along(iden.2ps.20br.l), function(i) {
  coll.plot.fx(iden.2ps.20br.l[[i]], mod = "2ps", 
               names(iden.2ps.20br.l)[i], 
               max(iden.2ps.20br.l[[i]]["collinearity"]))
})
#####

# Extract optimized pars from modfit output
#####
## bulk 14c only
# 0-10
pars.fit.2ps.10b <- lapply(mod.fits.2ps.10b, "[[", 1)
names(pars.fit.2ps.10b) <- names(pars.i.2ps)[ix.10]
# # 20-30
# pars.fit.2ps.30b <- lapply(mod.fits.2ps.30b, "[[", 1)
# names(pars.fit.2ps.30b) <- names(pars.i.2ps)[ix.30]

## resp + bulk 14c
# 0-10
pars.fit.2ps.10br <- lapply(mod.fits.2ps.10br, "[[", 1)
names(pars.fit.2ps.10br) <- names(pars.i.2ps)[ix.10]
# 10-20 w/ lag = 5y
pars.fit.2ps.20br.l <- lapply(mod.fits.2ps.20br.l, "[[", 1)
names(pars.fit.2ps.20br.l) <- names(pars.i.2ps)[ix.20]
# # 20-30
# pars.fit.2ps.30br <- lapply(mod.fits.2ps.30br, "[[", 1)
# names(pars.fit.2ps.30br) <- names(pars.i.2ps)[ix.30]

## bulk 14c + stocks
# 0-10
pars.fit.2ps.10bs <- lapply(mod.fits.2ps.10bs, "[[", 1)
names(pars.fit.2ps.10bs) <- names(pars.i.2ps)[ix.10]
# # 20-30
# pars.fit.2ps.30br <- lapply(mod.fits.2ps.30br, "[[", 1)
# names(pars.fit.2ps.30br) <- names(pars.i.2ps)[ix.30]

## resp, bulk 14c, stocks
# 0-10
pars.fit.2ps.10rbs <- lapply(mod.fits.2ps.10rbs, "[[", 1)
names(pars.fit.2ps.10rbs) <- names(pars.i.2ps)[ix.10]
# 10-20
pars.fit.2ps.20rbs <- lapply(mod.fits.2ps.20rbs, "[[", 1)
names(pars.fit.2ps.20rbs) <- names(pars.i.2ps)[ix.20]
# 10-20 w/ lag = 12y
pars.fit.2ps.20rbs.l <- lapply(mod.fits.2ps.20rbs.l, "[[", 1)
names(pars.fit.2ps.20rbs.l) <- names(pars.i.2ps)[ix.20]
# # 20-30
# pars.fit.2ps.30br <- lapply(mod.fits.2ps.30br, "[[", 1)
# names(pars.fit.2ps.30br) <- names(pars.i.2ps)[ix.30]
#####

# SOC stocks
#####
# w/o stock constraint
mod.socs.2ps.10b.ls <- lapply(seq_along(pars.fit.2ps.10b), function(i) {
  soc.fx("2ps", pars.fit.2ps.10b[[i]], in.est[[i]])
})
names(mod.socs.2ps.10b.ls) <- names(pars.fit.2ps.10b)
mod.socs.2ps.10br.ls <- lapply(seq_along(pars.fit.2ps.10br), function(i) {
  soc.fx("2ps", pars.fit.2ps.10br[[i]], in.est[[i]])
})
names(mod.socs.2ps.10br.ls) <- names(pars.fit.2ps.10br)
socs.2ps.10br.ls <- mapply(cbind,
                           csoc.19.0_30[ix.10], 
                           lapply(mod.socs.2ps.10br.ls, colSums), 
                           SIMPLIFY = FALSE)
# w/ stock constraint
mod.socs.2ps.10bs.ls <- lapply(seq_along(pars.fit.2ps.10bs), function(i) {
  soc.fx("2ps", pars.fit.2ps.10bs[[i]], in.est[[i]])
})
names(mod.socs.2ps.10bs.ls) <- names(pars.fit.2ps.10bs)
mod.socs.2ps.10rbs.ls <- lapply(seq_along(pars.fit.2ps.10rbs), function(i) {
  soc.fx("2ps", pars.fit.2ps.10rbs[[i]], in.est[[i]])
})
names(mod.socs.2ps.10rbs.ls) <- names(pars.fit.2ps.10rbs)
mod.socs.2ps.20rbs.ls <- lapply(seq_along(pars.fit.2ps.20rbs), function(i) {
  soc.fx("2ps", pars.fit.2ps.20rbs[[i]], in.est[ix.20][[i]])
})
names(mod.socs.2ps.20rbs.ls) <- names(pars.fit.2ps.20rbs)
socs.2ps.10rbs.ls <- mapply(cbind,
                           csoc.19.0_30[ix.10], 
                           lapply(mod.socs.2ps.10rbs.ls, colSums), 
                           SIMPLIFY = FALSE)

## make df for plotting
# resp + bulk, w/ and w/o stocks
mod.socs.2ps.10brrbs.df <- rbind(mod.socs.df.fx("2ps w/o stock", mod.socs.2ps.10br.ls, c("fast", "slow"))
                                 ,mod.socs.df.fx("2ps w/ stock", mod.socs.2ps.10rbs.ls, c("fast", "slow"))
                                 ,data.frame(SOC = unlist(lapply(csoc.19.0_30[ix.10], "[[", 4)),
                                             PMeco_depth = paste0(
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 1)),
                                               "_",
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 2)),
                                               "-",
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 3))),
                                             Model = "measured",
                                             pool = "total")
                                 )
# bulk + stock, vs. resp, bulk, + stock
mod.socs.2ps.10bsrbs.df <- rbind(mod.socs.df.fx("2ps bulk + stock only", mod.socs.2ps.10bs.ls, c("fast", "slow"))
                                 ,mod.socs.df.fx("2ps bulk, resp, + stock", mod.socs.2ps.10rbs.ls, c("fast", "slow"))
                                 ,data.frame(SOC = unlist(lapply(csoc.19.0_30[ix.10], "[[", 4)),
                                             PMeco_depth = paste0(
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 1)),
                                               "_",
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 2)),
                                               "-",
                                               unlist(lapply(csoc.19.0_30[ix.10], "[[", 3))),
                                             Model = "measured",
                                             pool = "total")
                                 )


## plot
mod.socs.2ps.10brrbs.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
mod.socs.2ps.10bsrbs.df %>%
  mutate(PM = substr(PMeco_depth, 1, 2),
         eco = factor(substr(PMeco_depth, 3, 4), levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(pool, SOC, fill = Model)) +
  geom_col(position = position_dodge()) +
  facet_grid(rows = vars(eco), cols = vars(PM)) +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
#####

### Summarize optimized par data for plotting
## bulk 14c only
# 0-10
pars.fit.2ps.10b.df <- par.fit.df.fx(mod = "2ps",
                                     pars.fit = pars.fit.2ps.10b,
                                     pars.i = pars.i.2ps[ix.10])
# # 20-30
# pars.fit.2ps.30b.df <- par.fit.df.fx(mod = "2ps",
#                                       pars.fit = pars.fit.2ps.30b,
#                                       pars.i = pars.i.2ps[ix.30])

## resp + bulk 14c
# 0-10
pars.fit.2ps.10br.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.10br,
                                       pars.i = pars.i.2ps[ix.10])
# 10-20
pars.fit.2ps.20br.l.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.20br.l,
                                       pars.i = pars.i.2ps[ix.10])
# # 20-30
# pars.fit.2ps.30br.df <- par.fit.df.fx(mod = "2ps",
#                                        pars.fit = pars.fit.2ps.30br,
#                                        pars.i = pars.i.2ps[ix.30])

## bulk 14c + stocks
# 0-10
pars.fit.2ps.10bs.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.10bs,
                                       pars.i = pars.i.2ps[ix.10])
# # 20-30
# pars.fit.2ps.30br.df <- par.fit.df.fx(mod = "2ps",
#                                        pars.fit = pars.fit.2ps.30br,
#                                        pars.i = pars.i.2ps[ix.30])

## resp, bulk, stocks
# 0-10
pars.fit.2ps.10rbs.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.10rbs,
                                       pars.i = pars.i.2ps[ix.10])
# 10-20
pars.fit.2ps.20rbs.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.20rbs,
                                       pars.i = pars.i.2ps[ix.20])
# w/ lag
pars.fit.2ps.20rbs.l.df <- par.fit.df.fx(mod = "2ps",
                                       pars.fit = pars.fit.2ps.20rbs.l,
                                       pars.i = pars.i.2ps[ix.20])
# # 20-30
# pars.fit.2ps.30br.df <- par.fit.df.fx(mod = "2ps",
#                                        pars.fit = pars.fit.2ps.30br,
#                                        pars.i = pars.i.2ps[ix.30])

### Par fits
par.plot.fx(mod = "2ps bulk 14c",
            depth = "0-10",
            par.df = pars.fit.2ps.10b.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp + bulk 14c",
            depth = "0-10",
            par.df = pars.fit.2ps.10br.df,
            initial = FALSE)
par.plot.fx(mod = "2ps bulk 14c + stocks",
            depth = "0-10",
            par.df = pars.fit.2ps.10bs.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp, bulk, stocks",
            depth = "0-10",
            par.df = pars.fit.2ps.10rbs.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp, bulk, stocks",
            depth = "10-20",
            par.df = pars.fit.2ps.20rbs.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp, bulk, stocks",
            depth = "10-20",
            par.df = pars.fit.2ps.20rbs.l.df,
            initial = FALSE)
par.plot.fx(mod = "2ps resp, bulk",
            depth = "10-20",
            par.df = pars.fit.2ps.20br.l.df,
            initial = FALSE)
# par.plot.fx(mod = "2ps bulk 14c",
#             depth = "20-30",
#             par.df = pars.fit.2ps.30b.df,
#             initial = FALSE)
# par.plot.fx(mod = "2ps resp + bulk 14c",
#             depth = "20-30",
#             par.df = pars.fit.2ps.30br.df,
#             initial = FALSE)

### Fit models with optimized pars
## bulk 14C only
# 0-10
Twops.10b.fits <- lapply(seq_along(pars.fit.2ps.10b), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.10b[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.10b.fits) <- names(pars.fit.2ps.10b)
# # 20-30
# Twops.30b.fits <- lapply(seq_along(pars.fit.2ps.30b), function(i) {
#   tryCatch(
#     par.fx(pars.fit.2ps.30b[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
#     error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
# })
# names(Twops.30b.fits) <- names(pars.fit.2ps.30b)

## resp + bulk 14C
# 0-10
Twops.10br.fits <- lapply(seq_along(pars.fit.2ps.10br), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.10br[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.10br.fits) <- names(pars.fit.2ps.10br)
# 10-20
Twops.20br.l.fits <- lapply(seq_along(pars.fit.2ps.20br.l), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.20br.l[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.20br.l.fits) <- names(pars.fit.2ps.20br.l)
# # 20-30
# Twops.30br.fits <- lapply(seq_along(pars.fit.2ps.30br), function(i) {
#   tryCatch(
#     par.fx(pars.fit.2ps.30br[[i]], in.est[ix.30][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
#     error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
# })
# names(Twops.30br.fits) <- names(pars.fit.2ps.30br)

## bulk 14C + stocks
# 0-10
Twops.10bs.fits <- lapply(seq_along(pars.fit.2ps.10bs), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.10bs[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.10bs.fits) <- names(pars.fit.2ps.10bs)

## resp, bulk, stocks
# 0-10
Twops.10rbs.fits <- lapply(seq_along(pars.fit.2ps.10rbs), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.10rbs[[i]], in.est[ix.10][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.10rbs.fits) <- names(pars.fit.2ps.10rbs)
# 10-20
Twops.20rbs.fits <- lapply(seq_along(pars.fit.2ps.20rbs), function(i) {
  tryCatch(
    par.fx(pars.fit.2ps.20rbs[[i]], in.est[ix.20][[i]], verbose = FALSE, mod = "2ps", pass = FALSE),
    error = function (e) {cat("ERROR :", conditionMessage(e), "\n")})
})
names(Twops.20rbs.fits) <- names(pars.fit.2ps.20rbs)

# # remove null entries
# Twops.10b.fits <- Filter(Negate(is.null), Twops.10b.fits)
# Twops.30b.fits <- Filter(Negate(is.null), Twops.30b.fits)
# Twops.10br.fits <- Filter(Negate(is.null), Twops.10br.fits)
# Twops.30br.fits <- Filter(Negate(is.null), Twops.30br.fits)

# Look at role of resp constraint in fit
# lapply(seq_along(Twops.10b.fits), function(i) {
#   C14.2p.plot.fx(Twops.10b.fits[[i]],
#                  con.df = con.df.fx(names(Twops.10b.fits)[i]), 
#                  mod = "2ps bulk only",
#                  PMeco_depth = names(Twops.10b.fits)[i])
# })
lapply(seq_along(Twops.10br.fits), function(i) {
  C14.2p.plot.fx(Twops.10br.fits[[i]], 
                 con.df = con.df.fx(names(Twops.10br.fits)[i]), 
                 mod = "2ps bulk + resp",
                 PMeco_depth = names(Twops.10br.fits)[i])
})
# lapply(seq_along(Twops.10bs.fits), function(i) {
#   C14.2p.plot.fx(Twops.10bs.fits[[i]],
#                  con.df = con.df.fx(names(Twops.10bs.fits)[i]), 
#                  mod = "2ps bulk + stock",
#                  PMeco_depth = names(Twops.10bs.fits)[i])
# })
lapply(seq_along(Twops.10rbs.fits), function(i) {
  C14.2p.plot.fx(Twops.10rbs.fits[[i]], 
                 con.df = con.df.fx(names(Twops.10rbs.fits)[i]), 
                 mod = "bulk, resp, stock",
                 PMeco_depth = names(Twops.10rbs.fits)[i])
})
# 10-20
lapply(seq_along(Twops.20rbs.fits), function(i) {
  C14.2p.plot.fx(Twops.20rbs.fits[[i]], 
                 con.df = con.df.fx(names(Twops.20rbs.fits)[i]), 
                 mod = "bulk, resp, stock",
                 PMeco_depth = names(Twops.20rbs.fits)[i])
})
# lapply(seq_along(Twops.30b.fits), function(i) {
#   C14.2p.plot.fx(Twops.30b.fits[[i]],
#                  con.df = con.df.fx(names(Twops.30b.fits)[i]), 
#                  mod = "2ps bulk only",
#                  PMeco_depth = names(Twops.30b.fits)[i])
# })
# lapply(seq_along(Twops.30br.fits), function(i) {
#   C14.2p.plot.fx(Twops.30br.fits[[i]], 
#                  con.df = con.df.fx(names(Twops.30br.fits)[i]), 
#                  mod = "2ps bulk + resp",
#                  PMeco_depth = names(Twops.30br.fits)[i])
# })

## Show role of resp in constraining models
# GRwf 0-10
Twop.fit.plot.fx(Twops.10bs.fits[which(names(Twops.10bs.fits) == "GRwf_0-10")], 
                 "2ps 0-10cm, bulk 14c + stock", 
                 Twops.10rbs.fits[which(names(Twops.10rbs.fits) == "GRwf_0-10")],
                 "2ps 0-10cm, resp & bulk 14c + stock")
# BSrf 0-10
Twop.fit.plot.fx(Twops.10br.fits[which(names(Twops.10br.fits) == "BSrf_0-10")], 
                 "2ps 0-10cm w/ resp", 
                 Twops.10b.fits[which(names(Twops.10b.fits) == "BSrf_0-10")],
                 "2ps 0-10cm w/o resp")
Twop.fit.plot.fx(Twops.10rbs.fits[which(names(Twops.10rbs.fits) == "BSrf_0-10")],
                 "2ps 0-10cm w/ resp, bulk, stocks",
                 Twops.10bs.fits[which(names(Twops.10bs.fits) == "BSrf_0-10")],
                 "2ps 0-10cm w/o resp (bulk + stocks only)")
Twop.fit.plot.fx(Twops.10rbs.fits[which(names(Twops.10rbs.fits) == "BSwf_10-20")],
                 "Basalt/cool 10-20",
                 Twops.10rbs.fits[which(names(Twops.10rbs.fits) == "GRwf_10-20")],
                 "Granite/cool 10-20")

# compare resp + bulk fits w/ and w/o stocks
Twop.fit.plot.fx(Twops.10rbs.fits, 
                 "2ps 0-10cm w/ resp, bulk, stocks", 
                 Twops.10br.fits,
                 "2ps 0-10cm w/ resp + bulk, no stock")
# compare resp + bulk fits w/ and w/o stocks
Twop.fit.plot.fx(Twops.10bs.fits, 
                 "2ps 0-10cm, bulk 14c + stock", 
                 Twops.10rbs.fits,
                 "2ps 0-10cm, resp & bulk 14c + stock")
# compare BSwf and GRwf 10-20
BSGRwf20.con.df <- cbind(rbind(con.df.fx("BSwf_10-20"), con.df.fx("GRwf_10-20")),
                               pm = factor(rep(c("basalt", "granite"), each = c(11))))
BSGRwf20.con.df <- BSGRwf20.con.df[-which(BSGRwf20.con.df$Year == 2009.5), ]
ANGRwf20.con.df <- cbind(rbind(con.df.fx("BSwf_10-20"), con.df.fx("GRwf_10-20")),
                               pm = factor(rep(c("basalt", "granite"), each = c(11))))
BSGRwf20.con.df <- BSGRwf20.con.df[-which(BSGRwf20.con.df$Year == 2009.5), ]
atm.14c2 <- Twops.20rbs.fits$`BSwf_10-20`[Twops.20rbs.fits$`BSwf_10-20`$years >= 1950 & Twops.20rbs.fits$`BSwf_10-20`$pool == "atm", ]
# plot
p <- rbind(Twops.20rbs.fits$`BSwf_10-20`,
      Twops.20rbs.fits$`GRwf_10-20`) %>%
  mutate(pm = rep(c("basalt", "granite"), 
                  each = nrow(Twops.20rbs.fits$`BSwf_10-20`))) %>%
  filter(pool == "bulk C" | pool == "respiration") %>%
  ggplot(., aes(years, d14C)) +
  geom_path(data = atm.14c2) +
  geom_path(aes(linetype = pool, color = pm)) +
  geom_point(data = BSGRwf20.con.df, 
             aes(Year, d14c, color = pm, shape = pool), 
             size = 2.5,
             position = position_dodge(width = 1)) +
  scale_color_manual(
    name = "Parent material",
    values = c("basalt" = "red",
               "granite" = "darkgray")) +
  scale_shape_manual(
    name = "",
    values = c("bulk C" = 16,
               "respiration" = 1)) +
  scale_linetype_manual(
   name = "Pool",
   values = c("bulk C" = 1,
              "respiration" = 2)) +
  scale_x_continuous(limits = c(1950, 2022)) +
  xlab("Year") +
  ylab(expression(''*Delta*''^14*'C (‰)')) +
  theme_bw() +
  theme(panel.grid = element_blank())
ggsave("sra.2ps.BSGRwf20.pdf", p, dpi = 300, width = 6, height = 5, units = "in")
# compare resp + bulk fits w/ and w/o stocks
Twop.fit.plot.fx(Twops.20br.l.fits, 
                 "2ps 0-10cm, bulk 14c + stock", 
                 Twops.20rbs.fits,
                 "2ps 0-10cm, resp & bulk 14c + stock")
#####

# ages and transit times
#####
# 2ps
SA.2ps.20.rbs.ls <- lapply(seq_along(pars.fit.2ps.20rbs), function(i) {
  ks <- pars.fit.2ps.20rbs[[i]][1:2]
  tc <- pars.fit.2ps.20rbs[[i]][3]
  In <- in.est[ix.20][[i]]
  A <- diag(-ks)
  A[2, 1] <- tc * ks[1]
  return(systemAge(A = A, u = c(In, 0)))
})
names(SA.2ps.20.rbs.ls) <- names(pars.fit.2ps.20rbs)
lapply(SA.2ps.20.rbs.ls, "[[", 1)

## Transit time
# 2ps
TT.2ps.20.rbs.ls <- lapply(seq_along(pars.fit.2ps.20rbs), function(i) {
  ks <- pars.fit.2ps.20rbs[[i]][1:2]
  tc <- pars.fit.2ps.20rbs[[i]][3]
  In <- in.est[ix.20][[i]]
  A <- diag(-ks)
  A[2, 1] <- tc * ks[1]
  return(transitTime(A = A, u = c(In, 0)))
})
names(TT.2ps.20.rbs.ls) <- names(pars.fit.2ps.20rbs)
lapply(TT.2ps.20.rbs.ls, "[[", 1)
# 0-10
TT.MA.2ps.10.rbs.ls <- lapply(seq_along(pars.fit.2ps.10rbs), function(i) {
  ks <- pars.fit.2ps.10rbs[[i]][1:2]
  tc <- pars.fit.2ps.10rbs[[i]][3]
  In <- in.est[ix.10][[i]]
  A <- diag(-ks)
  A[2, 1] <- tc * ks[1]
  TT <- transitTime(A = A, u = c(In, 0))
  Age <- systemAge(A = A, u = c(In, 0))
  return(list(TT = TT$meanTransitTime, Age = Age$meanSystemAge))
})
names(TT.MA.2ps.10.rbs.ls) <- names(pars.fit.2ps.10rbs)
lapply(TT.MA.2ps.10.rbs.ls, unlist)
# 
ageD.2ps.10.rbs.ls <- lapply(seq_along(pars.fit.2ps.10rbs), function(i) {
  ks <- pars.fit.2ps.10rbs[[i]][1:2]
  tc <- pars.fit.2ps.10rbs[[i]][3]
  In <- in.est[ix.10][[i]]
  A <- diag(-ks)
  A[2, 1] <- tc * ks[1]
  return(systemAge(A = A, u = c(In, 0)))
})
names(ageD.2ps.10.rbs.ls) <- names(pars.fit.2ps.10rbs)
```

```{r modFit-2p-comparison}
# compare output of 2pp and 2ps model fits
merge(ssr.2pp.df, ssr.2ps.df, by = "PMeco_depth", suffixes = c("_2pp", "_2ps")) %>%
  mutate(ssr_2pp = round(ssr_2pp, 1),
         ssr_2ps = round(ssr_2ps, 1),
         dif = ssr_2pp - ssr_2ps)
merge(var_ms.2pp.df,
      var_ms.2ps.df,
      by = c("PMeco_depth", "var"),
      suffixes = c("_2pp", "_2ps")) %>%
  mutate(var_ms_2pp = round(var_ms_2pp, 4),
         var_ms_2ps = round(var_ms_2ps, 4),
         dif = var_ms_2pp - var_ms_2ps)

## Plot
# SSR, PM
rbind(ssr.2pp.df, ssr.2ps.df) %>%
  mutate(mod = rep(c("2pp", "2ps"), each = nrow(ssr.2pp.df)),
         PM = substr(PMeco_depth, 1, 2),
         eco = substr(PMeco_depth, 3, 4)) %>%
  group_by(PM, mod) %>%
  summarize(mean.ssr = mean(ssr), sd = sd(ssr)) %>%
  mutate(err_u = mean.ssr + sd/sqrt(3),
         err_l = mean.ssr - sd/sqrt(3)) %>%
  ggplot(., aes(mod, mean.ssr, fill = PM)) +
  geom_col(position = "dodge") +
  geom_errorbar(
    aes(ymax = err_u, ymin = err_l), 
    position = position_dodge(width = .9),
    width = .3) +
  scale_fill_manual(name = "Parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ PM) +
  ggtitle("SSR 2-pool models 0-10 cm") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# SSR, eco
rbind(ssr.2pp.df, ssr.2ps.df) %>%
  mutate(mod = rep(c("2pp", "2ps"), each = nrow(ssr.2pp.df)),
         PM = substr(PMeco_depth, 1, 2),
         eco = substr(PMeco_depth, 3, 4)) %>%
  group_by(eco, mod) %>%
  summarize(mean.ssr = mean(ssr), sd = sd(ssr)) %>%
  mutate(err_u = mean.ssr + sd/sqrt(3),
         err_l = mean.ssr - sd/sqrt(3)) %>%
  ggplot(., aes(mod, mean.ssr, fill = eco)) +
  geom_col(position = "dodge") +
  geom_errorbar(
    aes(ymax = err_u, ymin = err_l), 
    position = position_dodge(width = .9),
    width = .3) +
  facet_wrap(. ~ eco) +
  ggtitle("SSR 2-pool models 0-10 cm (eco)") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# var_ms, PM
rbind(var_ms.2pp.df, var_ms.2ps.df) %>%
  mutate(mod = rep(c("2pp", "2ps"), each = nrow(var_ms.2pp.df)),
         PM = substr(PMeco_depth, 1, 2),
         eco = substr(PMeco_depth, 3, 4)) %>%
  group_by(var, PM, mod) %>%
  summarize(mean.var_ms = mean(var_ms), sd = sd(var_ms)) %>%
  mutate(err_u = mean.var_ms + sd/sqrt(3),
         err_l = mean.var_ms - sd/sqrt(3)) %>%
  ggplot(., aes(mod, mean.var_ms, fill = PM)) +
  geom_col(position = "dodge") +
  geom_errorbar(
    aes(ymax = err_u, ymin = err_l), 
    position = position_dodge(width = .9),
    width = .3) +
  scale_fill_manual(name = "Parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ var, scales = "free") +
  ggtitle("Residual error 2-pool models 0-10 cm") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# var_ms, eco
rbind(var_ms.2pp.df, var_ms.2ps.df) %>%
  mutate(mod = rep(c("2pp", "2ps"), each = nrow(var_ms.2pp.df)),
         PM = substr(PMeco_depth, 1, 2),
         eco = substr(PMeco_depth, 3, 4)) %>%
  group_by(var, eco, mod) %>%
  summarize(mean.var_ms = mean(var_ms), sd = sd(var_ms)) %>%
  mutate(err_u = mean.var_ms + sd/sqrt(3),
         err_l = mean.var_ms - sd/sqrt(3)) %>%
  ggplot(., aes(mod, mean.var_ms, fill = eco)) +
  geom_col(position = "dodge") +
  geom_errorbar(
    aes(ymax = err_u, ymin = err_l), 
    position = position_dodge(width = .9),
    width = .3) +
  facet_wrap(. ~ var, scales = "free") +
  ggtitle("Residual error 2-pool models 0-10 cm") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```

```{r ages-tt-modFit}
## System age
# 2pp
SA.2pp.ls <- lapply(seq_along(pars.fit.2pp), function(i) {
  ks <- pars.fit.2pp[[i]][1:2]
  gam <- pars.fit.2pp[[i]][3]
  In <- in.fit.2pp[[i]]
  return(systemAge(, u = In))
})
names(SA.2pp.ls) <- names(pars.fit.2pp)
# 2pp gam = [.5, .95]
SA.2pp.p3.5.95.ls <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  ks <- pars.fit.2pp.p3.5.95[[i]][1:2]
  gam <- pars.fit.2pp.p3.5.95[[i]][3]
  In <- in.fit.2pp.p3.5.95[[i]]
  return(systemAge(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(SA.2pp.p3.5.95.ls) <- names(pars.fit.2pp.p3.5.95)
# 2ps
SA.2ps.ls <- lapply(seq_along(pars.fit.2ps), function(i) {
  ks <- pars.fit.2ps[[i]][1:2]
  gam <- pars.fit.2ps[[i]][3]
  In <- in.fit.2ps[[i]]
  return(systemAge(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(SA.2ps.ls) <- names(pars.fit.2ps)

## Transit time
# 2pp
TT.2pp.ls <- lapply(seq_along(pars.fit.2pp), function(i) {
  ks <- pars.fit.2pp[[i]][1:2]
  gam <- pars.fit.2pp[[i]][3]
  In <- in.fit.2pp[[i]]
  return(transitTime(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(TT.2pp.ls) <- names(pars.fit.2pp)
# 2pp gam = [.5, .95]
TT.2pp.p3.5.95.ls <- lapply(seq_along(pars.fit.2pp.p3.5.95), function(i) {
  ks <- pars.fit.2pp.p3.5.95[[i]][1:2]
  gam <- pars.fit.2pp.p3.5.95[[i]][3]
  In <- in.fit.2pp.p3.5.95[[i]]
  return(transitTime(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(TT.2pp.p3.5.95.ls) <- names(pars.fit.2pp.p3.5.95)
# 2ps
TT.2ps.ls <- lapply(seq_along(pars.fit.2ps), function(i) {
  ks <- pars.fit.2ps[[i]][1:2]
  gam <- pars.fit.2ps[[i]][3]
  In <- in.fit.2ps[[i]]
  return(transitTime(A = -1 * diag(ks), u = c(In * gam, In * (1 - gam))))
})
names(TT.2ps.ls) <- names(pars.fit.2ps)
```

```{r SA-TT-comp}
# compare ages and transit times among the two model structures
SA.2p.ls <- list(SA.2pp.ls, SA.2ps.ls, SA.2pp.p3.5.95.ls)
SA.df <- bind_rows(
  lapply(SA.2p.ls, function(ls) {
    lapply(seq_along(ls), function(i) {
      data.frame(age = c(ls[[i]][["meanSystemAge"]],
                         ls[[i]][["meanPoolAge"]]),
                 component = c("system", "fast pool", "slow pool"))
    })
  })
)
SA.df$PMeco_depth <- rep(names(SA.2pp.ls), each = 3, times = length(SA.2p.ls))
SA.df$Model <- rep(c("2pp", "2ps", "2pp [.5, .95]"), each = 27)
TT.2p.ls <- list(TT.2pp.ls, TT.2ps.ls, TT.2pp.p3.5.95.ls)
TT.df <- bind_rows(
  lapply(TT.2p.ls, function(ls) {
    lapply(seq_along(ls), function(i) {
     data.frame(age = ls[[i]][["meanTransitTime"]],
                component = "transit")
    })
  })
)
TT.df$PMeco_depth <- rep(names(TT.2pp.ls), times = length(TT.2p.ls))
TT.df$Model <- rep(c("2pp", "2ps", "2pp [.5, .95]"), each = 9)
SA.TT.df <- rbind(SA.df, TT.df)
SA.TT.df$PM <- substr(SA.TT.df$PMeco_depth, start = 1, stop = 2)
SA.TT.df$eco <- substr(SA.TT.df$PMeco_depth, start = 3, stop = 4)

## Plot ages and transit times
# by PM
SA.TT.df %>%
  select(!c(PMeco_depth, eco)) %>%
  group_by(component, PM, Model) %>%
  summarize_all(list(mean_age = mean, sd = sd)) %>%
  mutate(err_u = mean_age + sd,
         err_l = mean_age - sd) %>%
  ggplot(., aes(Model, mean_age, fill = PM)) +
  geom_col(position = "dodge") +
  # geom_errorbar(
  #   aes(ymax = err_u, ymin = err_l), 
  #   position = position_dodge(width = .9),
  #   width = .3) +
  scale_fill_manual(name = "Parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ component, scales = "free") +
  ylab("mean age") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
# by eco
SA.TT.df %>%
  select(!c(PMeco_depth, PM)) %>%
  group_by(component, eco, Model) %>%
  summarize_all(list(mean_age = mean, sd = sd)) %>%
  mutate(err_u = mean_age + sd,
         err_l = mean_age - sd) %>%
  ggplot(., aes(Model, mean_age, fill = eco)) +
  geom_col(position = "dodge") +
  # geom_errorbar(
  #   aes(ymax = err_u, ymin = err_l),
  #   position = position_dodge(width = .9),
  #   width = .3) +
  facet_wrap(. ~ component, scales = "free") +
  ylab("mean age") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```

### Bayesian parameter estimation (MCMC)

```{r MCMC-fits, eval = FALSE}
# the following .RData files are generated by script "sra-ts/source/sra-ts-14c-mcmc-bayes.R"
load(file = "../data/derived/bayes-par-fit-2020-11-06/bayes_fit_2pp_0-10_5000iter.RData")
load(file = "../data/derived/bayes-par-fit-2020-11-17/bayes_fit_2ps_0-10_5000iter.RData")

# # plot parameter convergence
# lapply(bayes_fit_2pp_0_10, plot)
# lapply(bayes_fit_2ps_0_10, plot)

# plot collinearity
iter <- 5000
lapply(bayes_fit_2pp_0_10, pairs, nsample = floor(iter/4))
lapply(bayes_fit_2ps_0_10, pairs, nsample = floor(iter/4))

## look at model performance
pars.bayes.df.fx <- function(mod, pars.bayes, pars.fit) {
  bind_rows(lapply(seq_along(pars.bayes), function(i) {
    ix <- match(unique(pars.bayes[[i]][["pars"]][, 1]), pars.bayes[[i]][["pars"]][, 1])
    df <- data.frame(k1 = pars.bayes[[i]][["pars"]][ix, 1],
                     k2 = pars.bayes[[i]][["pars"]][ix, 2],
                     p3 = pars.bayes[[i]][["pars"]][ix, 3])
    df <- cbind(df,
                PMeco_depth = rep(names(pars.fit)[i], length(ix)),
                mod = rep(mod, length(ix)))
    df <- cbind(df, 
                PM = factor(substr(df$PMeco_depth, 1, 2)),
                eco = factor(substr(df$PMeco_depth, 3, 4), levels = c("pp", "wf", "rf")))
    return(df)
  }))
}
pars.bayes.2pp.df <- pars.bayes.df.fx("2pp", bayes_fit_2pp_0_10, pars.fit.2pp)
pars.bayes.2ps.df <- pars.bayes.df.fx("2ps", bayes_fit_2ps_0_10, pars.fit.2ps)

# # linear fits
# summary(lm(k2 ~ PM, pars.bayes.2pp.df))
# summary(lm(k2 ~ eco, pars.bayes.2pp.df))
# summary(lm(k1 ~ PM, pars.bayes.2pp.df))
# summary(lm(k1 ~ eco, pars.bayes.2pp.df))
# summary(lm(p3 ~ PM, pars.bayes.2pp.df))
# summary(lm(p3 ~ eco, pars.bayes.2pp.df))

# best par set
bestPars.bayes.ls <- lapply(bayes_fit_2pp_0_10, function(x) {
  round(data.frame(k1 = x$bestpar[1],
                   k2 = x$bestpar[2],
                   gam = x$bestpar[3]),
        4)
})
bestPars.bayes.df <- cbind(PM = rep(c("AN", "BS", "GR"), each = 3),
                           eco = rep(c("pp", "rf", "wf"), 3),
                           depth = rep("0-10", 9),
                           bind_rows(bestPars.bayes.ls))

# summarize by PM
pars.bayes.PM <- bestPars.bayes.df %>%
  select(!c(eco, depth)) %>%
  group_by(PM) %>%
  summarize_all(list(mean = mean, sd = sd)) %>%
  mutate_if(is.numeric, format, digits = 3)
# summarize by ECO
pars.bayes.eco <- bestPars.bayes.df %>%
  select(!c(PM, depth)) %>%
  group_by(eco) %>%
  summarize_all(list(mean = mean, sd = sd)) %>%
  mutate_if(is.numeric, format, digits = 3)

# plot best pars
bestPars.bayes.df %>%
  pivot_longer(!(PM:depth), names_to = "par", values_to = "value") %>%
  mutate(PM = factor(PM),
         eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(par, value, color = PM, shape = eco)) +
  geom_jitter(size = 4) +
  scale_color_manual(name = "parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ par, scales = "free") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())

# plot accepted pars by PM and then by eco
pars.bayes.df %>%
  pivot_longer(!c(PM, eco, PMeco_depth), names_to = "par", values_to = "value") %>%
  mutate(PM = factor(PM),
         eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(par, value, fill = PM)) +
  geom_boxplot() +
  scale_fill_manual(name = "parent material",
                    labels = c("AN" = "andesite",
                               "BS" = "basalt",
                               "GR" = "granite"),
                    values = c("AN" = "blue", 
                               "BS" = "red", 
                               "GR" = "darkgray")) +
  facet_wrap(. ~ par, scales = "free") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
pars.bayes.df %>%
  pivot_longer(!c(PM, eco, PMeco_depth), names_to = "par", values_to = "value") %>%
  mutate(PM = factor(PM),
         eco = factor(eco, levels = c("pp", "wf", "rf"))) %>%
  ggplot(., aes(par, value, fill = eco)) +
  geom_boxplot() +
  facet_wrap(. ~ par, scales = "free") +
  theme_bw() +
  theme(panel.grid.minor = element_blank())
```